解析Visual C# 7.2中的private protected訪問修飾符


去年12月份,隨着Visual Studio 2017 Update 15.5的發布,Visual C#迎來了它的最新版本:7.2. 在這個版本中,有個讓人難以理解的新特性,就是private protected訪問修飾符(Access Modifier)。至此,C#語言的訪問修飾符有以下幾種:

  • private
  • protected
  • public
  • internal
  • internal protected
  • private protected

既然有了private和protected,那么private protected又是什么?它跟internal protected又有什么關系?本文簡單介紹一下。

private protected是怎么回事

在解釋private protected之前,首先讓我們回顧一下internal protected訪問修飾符。internal protected表示,相同程序集(Assembly)中的其它類型,或者當前類的子類,具有訪問該類中internal protected成員的能力,可以用下圖表示:

image

在上圖中:

  • 程序集A中的X類,可以訪問A類中的Method方法
  • 程序集A中的B類,可以重載A類中的Method方法,B類中的其它成員也可以訪問A類中的Method
  • 程序集B中的C類,可以重載A類中的Method方法,C類中的其它成員也可以通過base.Method()訪問A類中的Method
  • 程序集B中C類的Method方法重載了A類的Method方法,因此,internal關鍵字被去掉,於是,程序集B中的Y類,無法訪問C類中的Method方法

因此,internal protected表示internal或者protected。

然而,private protected表示,僅有相同程序集(Assembly)中繼承於當前類型的類,才能訪問該類中private protected成員。換句話說,private protected就是訪問者必須在相同程序集中(internal),同時還必須是被訪問類型的子類(protected),可以用下圖表示:

image

因此,private protected表示internal並且protected。

private protected何時使用

理論上講,private protected完善了C#語言的封裝性,提供了另一層級別的成員訪問保護,聽起來感覺讓人費解又沒什么用。那么,什么時候使用這個訪問修飾符呢?現假設你正在設計一個框架,其中有個類,它提供對象存儲功能,它的職責是保存給定的對象,而它的每一種實現都需要依賴於一個對象的序列化機制,比如:

public sealed class SerializationHelper
{
    public string Serialze(object s)
    {
        using (var memoryStream = new MemoryStream())
        {
            var serializer = new XmlSerializer(s.GetType());
            serializer.Serialize(memoryStream, s);
            return Encoding.UTF8.GetString(memoryStream.ToArray());
        }
    }
}

public abstract class DataStorage
{
    private readonly SerializationHelper serializer = new SerializationHelper();
    protected SerializationHelper Serializer => serializer;
    protected abstract void SaveObject(object obj);
}

public sealed class InMemoryDataStorage : DataStorage
{
    private readonly List<string> serializedData = new List<string>();
    protected override void SaveObject(object obj) 
        => serializedData.Add(Serializer.Serialze(obj));
}

上面的代碼中,SerializationHelper提供了一種將對象序列化成XML字符串的機制;DataStorage是所有對象數據存儲的基類,它當然也為其子類提供了一個訪問對象序列化器的方式。由於這個對象序列化器是提供給其子類調用的,因此,DataStorage中的Serializer屬性是protected的。最后,InMemoryDataStorage繼承了DataStorage,通過調用由基類提供的Serializer屬性,實現了SaveObject方法。

整個實現當然沒有問題。可是,通過審核所有類型的可見性,我們發現,我們不打算將SerializationHelper這個類暴露給外界,也就是不希望其它的程序集能夠直接訪問SerializationHelper類,於是,我們將它設置成internal的。也就是:

internal sealed class SerializationHelper
{
    public string Serialze(object s)
    {
        using (var memoryStream = new MemoryStream())
        {
            var serializer = new XmlSerializer(s.GetType());
            serializer.Serialize(memoryStream, s);
            return Encoding.UTF8.GetString(memoryStream.ToArray());
        }
    }
}

好了,問題來了,編譯器開始抱怨了,說SerializationHelper類的訪問級別比DataStorage.Serializer屬性的訪問級別要低:

image

道理顯而易見:DataStorage.Serializer屬性在DataStorage的子類中即可訪問,這個子類可以是在DataStorage所在的程序集中,也可以是在另一個程序集中。然而,這個屬性的依賴類型:SerializationHelper類,卻只能在DataStorage所在的程序集中才能被訪問。

於是,能量巨大的private protected閃亮登場。將DataStorage.Serializer屬性的訪問修飾符從protected改為private protected,問題就解決了:

internal sealed class SerializationHelper
{
    public string Serialze(object s)
    {
        using (var memoryStream = new MemoryStream())
        {
            var serializer = new XmlSerializer(s.GetType());
            serializer.Serialize(memoryStream, s);
            return Encoding.UTF8.GetString(memoryStream.ToArray());
        }
    }
}

public abstract class DataStorage
{
    private readonly SerializationHelper serializer = new SerializationHelper();
    private protected SerializationHelper Serializer => serializer;
    protected abstract void SaveObject(object obj);
}

public sealed class InMemoryDataStorage : DataStorage
{
    private readonly List<string> serializedData = new List<string>();
    protected override void SaveObject(object obj) 
        => serializedData.Add(Serializer.Serialze(obj));
}

不過,一旦使用了private protected訪問修飾符,DataStorage.Serializer屬性就只能在DataStorage所在的程序集的子類中訪問了。

private protected如何使用

private protected訪問修飾符是C# 7.2的新特性。自從使用Roslyn編譯器服務的C# 6.0開始,C#編譯器的版本更新就可以與.NET Framework和Visual Studio的發布分離開來了。這一點在C# 7.x(包括7.1和7.2)的發布中逐步顯現出來。在Visual Studio 2017的編譯高級選項中,開發人員可以很方便地選擇所需的C#版本:

image

如上圖所述,在C#項目上點右鍵,在項目屬性的Build標簽頁中,點擊Advanced按鈕,在Advanced Build Settings對話框中,通過Language version下拉框來選擇所需的C#語言版本。其中:

  • C# latest major version (default):C#最新的主版本,也就是與Visual Studio一起發布的主要版本,在VS2017上對應C# 7.0
  • C# latest minor version (latest):C#的最新版,通常通過VS2017的升級包獲得

要使用private protected訪問修飾符,則需要在此選擇C# latest minor version (latest),或者C# 7.2.

總結

本文對C# 7.2的新特性:private protected訪問修飾符進行了解析,並通過案例來說明它的應用場景以及Visual Studio 2017對於C#新特性的支持。


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM