解析Visual C# 7.2中的private protected访问修饰符
去年12月份,隨著Visual Studio 2017 Update 15.5的發(fā)布,Visual C#迎來了它的最新版本:7.2. 在這個版本中,有個讓人難以理解的新特性,就是private protected訪問修飾符(Access Modifier)。至此,C#語言的訪問修飾符有以下幾種:
private
protected
public
internal
internal protected
private protected
既然有了private和protected,那么private protected又是什么?它跟internal protected又有什么關(guān)系?本文簡單介紹一下。
private protected是怎么回事
在解釋private protected之前,首先讓我們回顧一下internal protected訪問修飾符。internal protected表示,相同程序集(Assembly)中的其它類型,或者當(dāng)前類的子類,具有訪問該類中internal protected成員的能力,可以用下圖表示:
在上圖中:
程序集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關(guān)鍵字被去掉,于是,程序集B中的Y類,無法訪問C類中的Method方法
因此,internal protected表示internal或者protected。
然而,private protected表示,僅有相同程序集(Assembly)中繼承于當(dāng)前類型的類,才能訪問該類中private protected成員。換句話說,private protected就是訪問者必須在相同程序集中(internal),同時還必須是被訪問類型的子類(protected),可以用下圖表示:
因此,private protected表示internal并且protected。
private protected何時使用
理論上講,private protected完善了C#語言的封裝性,提供了另一層級別的成員訪問保護,聽起來感覺讓人費解又沒什么用。那么,什么時候使用這個訪問修飾符呢?現(xiàn)假設(shè)你正在設(shè)計一個框架,其中有個類,它提供對象存儲功能,它的職責(zé)是保存給定的對象,而它的每一種實現(xiàn)都需要依賴于一個對象的序列化機制,比如:
| 1234567891011121314151617181920212223242526 | 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是所有對象數(shù)據(jù)存儲的基類,它當(dāng)然也為其子類提供了一個訪問對象序列化器的方式。由于這個對象序列化器是提供給其子類調(diào)用的,因此,DataStorage中的Serializer屬性是protected的。最后,InMemoryDataStorage繼承了DataStorage,通過調(diào)用由基類提供的Serializer屬性,實現(xiàn)了SaveObject方法。
整個實現(xiàn)當(dāng)然沒有問題。可是,通過審核所有類型的可見性,我們發(fā)現(xiàn),我們不打算將SerializationHelper這個類暴露給外界,也就是不希望其它的程序集能夠直接訪問SerializationHelper類,于是,我們將它設(shè)置成internal的。也就是:
| 123456789101112 | 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屬性的訪問級別要低:
道理顯而易見:DataStorage.Serializer屬性在DataStorage的子類中即可訪問,這個子類可以是在DataStorage所在的程序集中,也可以是在另一個程序集中。然而,這個屬性的依賴類型:SerializationHelper類,卻只能在DataStorage所在的程序集中才能被訪問。
于是,能量巨大的private protected閃亮登場。將DataStorage.Serializer屬性的訪問修飾符從protected改為private protected,問題就解決了:
| 1234567891011121314151617181920212223242526 | 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編譯器服務(wù)的C# 6.0開始,C#編譯器的版本更新就可以與.NET Framework和Visual Studio的發(fā)布分離開來了。這一點在C# 7.x(包括7.1和7.2)的發(fā)布中逐步顯現(xiàn)出來。在Visual Studio 2017的編譯高級選項中,開發(fā)人員可以很方便地選擇所需的C#版本:
如上圖所述,在C#項目上點右鍵,在項目屬性的Build標(biāo)簽頁中,點擊Advanced按鈕,在Advanced Build Settings對話框中,通過Language version下拉框來選擇所需的C#語言版本。其中:
C# latest major version (default):C#最新的主版本,也就是與Visual Studio一起發(fā)布的主要版本,在VS2017上對應(yīng)C# 7.0
C# latest minor version (latest):C#的最新版,通常通過VS2017的升級包獲得
要使用private protected訪問修飾符,則需要在此選擇C# latest minor version (latest),或者C# 7.2.
總結(jié)
本文對C# 7.2的新特性:private protected訪問修飾符進行了解析,并通過案例來說明它的應(yīng)用場景以及Visual Studio 2017對于C#新特性的支持。
原文地址:https://www.cnblogs.com/kenwoo/p/9333042.html
.NET社區(qū)新聞,深度好文,歡迎訪問公眾號文章匯總 http://www.csharpkit.com
總結(jié)
以上是生活随笔為你收集整理的解析Visual C# 7.2中的private protected访问修饰符的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 微软宣布公开预览Dev Spaces f
- 下一篇: Net Core集成Exceptionl