詳解C#中的接口屬性以及屬性訪問器的訪問限制
接口屬性
可以在接口上聲明屬性。以下是接口索引器訪問器的示例:
public interface ISampleInterface { // Property declaration: string Name { get; set; } }
接口屬性的訪問器不具有體。因此,訪問器的用途是指示屬性是否為讀寫、只讀或只寫。
在此例中,接口 IEmployee 具有讀寫屬性 Name 和只讀屬性 Counter。 Employee 類實(shí)現(xiàn) IEmployee 接口并使用這兩種屬性。程序讀取新雇員的姓名和雇員的當(dāng)前編號(hào),并顯示雇員姓名和計(jì)算所得的雇員編號(hào)。
可以使用屬性的完全限定名,它引用聲明成員的接口。例如:
string IEmployee.Name { get { return "Employee Name"; } set { } }
這稱為顯式接口實(shí)現(xiàn)(C# 編程指南)。例如,如果 Employee 類實(shí)現(xiàn)兩個(gè)接口 ICitizen 和 IEmployee,并且兩個(gè)接口都具有 Name 屬性,則需要顯式接口成員實(shí)現(xiàn)。即,如下屬性聲明:
string IEmployee.Name { get { return "Employee Name"; } set { } }
在 IEmployee 接口上實(shí)現(xiàn) Name 屬性,而下面的聲明:
string ICitizen.Name { get { return "Citizen Name"; } set { } }
在 ICitizen 接口上實(shí)現(xiàn) Name 屬性。
interface IEmployee { string Name { get; set; } int Counter { get; } } public class Employee : IEmployee { public static int numberOfEmployees; private string name; public string Name // read-write instance property { get { return name; } set { name = value; } } private int counter; public int Counter // read-only instance property { get { return counter; } } public Employee() // constructor { counter = ++counter + numberOfEmployees; } } class TestEmployee { static void Main() { System.Console.Write("Enter number of employees: "); Employee.numberOfEmployees = int.Parse(System.Console.ReadLine()); Employee e1 = new Employee(); System.Console.Write("Enter the name of the new employee: "); e1.Name = System.Console.ReadLine(); System.Console.WriteLine("The employee information:"); System.Console.WriteLine("Employee number: {0}", e1.Counter); System.Console.WriteLine("Employee name: {0}", e1.Name); } }
比如這里我們輸入:
210 Hazem Abolrous
則示例輸出
Enter number of employees: 210 Enter the name of the new employee: Hazem Abolrous The employee information: Employee number: 211 Employee name: Hazem Abolrous
限制訪問器可訪問性
屬性或索引器的 get 和 set 部分稱為“訪問器”。默認(rèn)情況下,這些訪問器具有相同的可見性或訪問級(jí)別:其所屬屬性或索引器的可見性或訪問級(jí)別。不過,有時(shí)限制對(duì)其中某個(gè)訪問器的訪問會(huì)很有用。通常是在保持 get 訪問器可公開訪問的情況下,限制 set 訪問器的可訪問性。例如:
private string name = "Hello"; public string Name { get { return name; } protected set { name = value; } }
在此示例中,名為 Name 的屬性定義了一個(gè) get 訪問器和一個(gè) set 訪問器。 get 訪問器接受該屬性本身的可訪問性級(jí)別(在此示例中為 public),而對(duì)于 set 訪問器,則通過對(duì)該訪問器本身應(yīng)用 protected 訪問修飾符來進(jìn)行顯式限制。
對(duì)訪問器的訪問修飾符的限制
對(duì)屬性或索引器使用訪問修飾符受以下條件的制約:
不能對(duì)接口或顯式接口成員實(shí)現(xiàn)使用訪問器修飾符。
僅當(dāng)屬性或索引器同時(shí)具有 set 和 get 訪問器時(shí),才能使用訪問器修飾符。這種情況下,只允許對(duì)其中一個(gè)訪問器使用修飾符。
如果屬性或索引器具有 override 修飾符,則訪問器修飾符必須與重寫的訪問器的訪問器(如果有的話)匹配。
訪問器的可訪問性級(jí)別必須比屬性或索引器本身的可訪問性級(jí)別具有更嚴(yán)格的限制。
重寫訪問器的訪問修飾符
在重寫屬性或索引器時(shí),被重寫的訪問器對(duì)重寫代碼而言,必須是可訪問的。此外,屬性/索引器和訪問器的可訪問性級(jí)別都必須與相應(yīng)的被重寫屬性/索引器和訪問器匹配。例如:
public class Parent { public virtual int TestProperty { // Notice the accessor accessibility level. protected set { } // No access modifier is used here. get { return 0; } } } public class Kid : Parent { public override int TestProperty { // Use the same accessibility level as in the overridden accessor. protected set { } // Cannot use access modifier here. get { return 0; } } }
實(shí)現(xiàn)接口
使用訪問器實(shí)現(xiàn)接口時(shí),訪問器不能具有訪問修飾符。但是,如果使用一個(gè)訪問器(如 get)實(shí)現(xiàn)接口,則另一個(gè)訪問器可以具有訪問修飾符,如下面的示例所示:
public interface ISomeInterface { int TestProperty { // No access modifier allowed here // because this is an interface. get; } } public class TestClass : ISomeInterface { public int TestProperty { // Cannot use access modifier here because // this is an interface implementation. get { return 10; } // Interface property does not have set accessor, // so access modifier is allowed. protected set { } } }
訪問器可訪問性域
如果對(duì)訪問器使用訪問某個(gè)修飾符,則訪問器的可訪問性域由該修飾符確定。
如果不對(duì)訪問器使用訪問修飾符,則訪問器的可訪問性域由屬性或索引器的可訪問性級(jí)別確定。
下面的示例包含三個(gè)類:BaseClass、DerivedClass 和 MainClass。每個(gè)類的 BaseClass、Name 和 Id 都有兩個(gè)屬性。該示例演示在使用限制性訪問修飾符(如 protected 或 private)時(shí),如何通過 BaseClass 的 Id 屬性隱藏 DerivedClass 的 Id 屬性。因此,向該屬性賦值時(shí),將調(diào)用 BaseClass 類中的屬性。將訪問修飾符替換為 public 將使該屬性可訪問。
該示例還演示 DerivedClass 的 Name 屬性的 set 訪問器上的限制性訪問修飾符(如 private 或 protected)如何防止對(duì)該訪問器的訪問,并在向它賦值時(shí)生成錯(cuò)誤。
public class BaseClass { private string name = "Name-BaseClass"; private string id = "ID-BaseClass"; public string Name { get { return name; } set { } } public string Id { get { return id; } set { } } } public class DerivedClass : BaseClass { private string name = "Name-DerivedClass"; private string id = "ID-DerivedClass"; new public string Name { get { return name; } // Using "protected" would make the set accessor not accessible. set { name = value; } } // Using private on the following property hides it in the Main Class. // Any assignment to the property will use Id in BaseClass. new private string Id { get { return id; } set { id = value; } } } class MainClass { static void Main() { BaseClass b1 = new BaseClass(); DerivedClass d1 = new DerivedClass(); b1.Name = "Mary"; d1.Name = "John"; b1.Id = "Mary123"; d1.Id = "John123"; // The BaseClass.Id property is called. System.Console.WriteLine("Base: {0}, {1}", b1.Name, b1.Id); System.Console.WriteLine("Derived: {0}, {1}", d1.Name, d1.Id); // Keep the console window open in debug mode. System.Console.WriteLine("Press any key to exit."); System.Console.ReadKey(); } }
輸出:
Base: Name-BaseClass, ID-BaseClass Derived: John, ID-BaseClass
相關(guān)文章
兩路歸并的數(shù)組與鏈表的實(shí)現(xiàn)方法
本篇文章對(duì)兩路歸并的數(shù)組與鏈表的實(shí)現(xiàn)方法進(jìn)行了分析介紹。需要的朋友參考下2013-05-05C# Winform實(shí)現(xiàn)導(dǎo)入和導(dǎo)出Excel文件
這篇文章主要為大家詳細(xì)介紹了C# Winform實(shí)現(xiàn)導(dǎo)入和導(dǎo)出Excel文件,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-12-12C#網(wǎng)頁(yè)跳轉(zhuǎn)方法總結(jié)
這篇文章主要介紹了C#網(wǎng)頁(yè)跳轉(zhuǎn)方法總結(jié)的相關(guān)資料,需要的朋友可以參考下2015-12-12C#連接Oracle數(shù)據(jù)庫(kù)的多種方法總結(jié)
最近小項(xiàng)目當(dāng)中要使用C#來連接Oracle數(shù)據(jù)庫(kù)來完成系統(tǒng)的操作,這篇文章主要給大家介紹了關(guān)于C#連接Oracle數(shù)據(jù)庫(kù)的多種方法,文中通過實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下2023-04-04詳解WPF如何在基礎(chǔ)控件上顯示Loading等待動(dòng)畫
這篇文章主要為大家詳細(xì)介紹了WPF如何在基礎(chǔ)控件上顯示Loading等待動(dòng)畫的效果,文中的示例代碼講解詳細(xì),具有一定的學(xué)習(xí)價(jià)值,需要的可以參考一下2023-04-04