(C#7.2)“私有保护”修饰符的用例是什么?

问题描述 投票:61回答:3

C# 7.2 introduces the private protected modifier

我总是保护对具有属性的字段的访问,允许通过Get / Set方法进行访问,因为我通常不希望我的对象的内部状态被我自己的类以外的任何东西修改。

我试图理解为什么C#语言团队添加了这个功能。经过广泛搜索谷歌,阅读和观看'什么是新的'媒体(我看过press releasedetailsvideo by Mads Torgerson),我仍然没有更聪明。

对我来说,这似乎允许开发人员打破Liskov替换原则,但这可能是因为我不明白为什么现在存在此功能。

我理解它是如何使用的,而不是为什么 - 请有人提供一个真实的使用示例而不是MSDN文档中的人为例子吗?

c# .net access-modifiers c#-7.2
3个回答
61
投票

在C#7.2之前,我们有protected internal修饰符。这实际上意味着受保护的OR内部,即 - 成员A可以被子类访问,也可以访问当前程序集中的任何类,即使该类不是类A的子类(因此放宽了“受保护”所暗示的限制)。

private protected真的意味着受保护和内部。也就是说 - 成员只能访问同一个程序集中的子类,而不能访问程序集之外的子类(因此“受保护”所暗示的限制变窄 - 变得更加严格)。如果在程序集中构建类的层次结构并且不希望其他程序集中的任何子类访问该层次结构的某些部分,那么这非常有用。

我们可以举例说,Jon Skeet提供了in comments。假设你上课了

public class MyClass {

}

并且您希望能够仅在当前程序集中继承它,但不希望允许直接实例化此类,除非在此类层次结构中。

可以使用内部构造函数实现仅在当前程序集内继承

public class MyClass {
    internal MyClass() {
    }
}

除了使用当前类层次结构之外,可以使用受保护的构造函数来实现防止直接实例化:

public class MyClass {
    protected MyClass() {
    }
}

要获得两者 - 你需要private protected构造函数:

public class MyClass {
    private protected MyClass() {
    }
}

8
投票

让我们假设您有一个名为SomeHelper的内部类,您希望将其用作公共抽象基类实现的一部分:

public abstract class Test
{
    // Won't compile because SomeHelper is internal.
    protected SomeHelper CreateHelper()
    {
        return new SomeHelper();
    }

    public int Func(int x)
    {
        var helper = CreateHelper();
        return helper.DoSomething(x);
    }
}

internal class SomeHelper
{
    public virtual int DoSomething(int x)
    {
        return -x;
    }
}

这将无法编译,因为您无法使用受保护的方法返回内部类型。你唯一的办法就是不要以这种方式使用SomeHelper,或者将SomeHelper公之于众。

(你可以使SomeHelper成为受保护的Test内部类,但是如果SomeHelper打算供其他不从基类派生的类使用的话,那就无法工作了。)

随着private protected功能的引入,您可以像这样声明CreateHelper()

private protected SomeHelper CreateHelper()
{
    return new SomeHelper();
}

现在它将编译,您不必暴露您的内部。


8
投票

对于双字访问修饰符,我有这个概念 - 第一个访问器与另一个组件相关,第二个访问器与定义它的组件相关。

保护内部

  • 在另一个程序集中受保护:只能在子类中访问。
  • 当前程序集中的内部:当前程序集中的每个人都可以访问。

私人保护

  • 另一个程序集中的私有:无法访问。
  • 在当前程序集中受保护:仅在子类中可访问。
© www.soinside.com 2019 - 2024. All rights reserved.