何为显式接口实现、隐式接口实现?简单概括,使用接口名作为方法名的前缀,这称为“显式接口实现”;传统的实现方式,称为“隐式接口实现”。下面给个例子。
IChineseGreeting接口,如果需要具备中文的问好行为,需继承此接口。
1 //IChineseGreeting.cs
2
3 interface IChineseGreeting
4
5 {
6
7 void SayHello();
8
9 }
10
11
12
13 ChinesePeople类实现IChineseGreeting接口。
14
15
16
17 //ChinesePeople.cs
18
19 class ChinesePeople:IChineseGreeting
20
21 {
22
23 public void SayHello()
24
25 {
26
27 Console.WriteLine("你好!");
28
29 }
30
31
32
33 }
这样的实现就是“隐式接口实现”,用public关键字作方法的访问修饰符,我们通常都是这么做的。
你可以这样调用:
ChinesePeople chinese = new ChinesePeople();
chinese.SayHello();
也可以这样调用
IChineseGreeting otherChinese = new ChinesePeople();
otherChinese.SayHello();
类和接口都能访问到SayHello()方法。
如果“显式接口实现”,就应该是这样。
1 //ChinesePeople.cs
2
3 class ChinesePeople:IChineseGreeting
4
5 {
6
7 void IChineseGreeting.SayHello() //注意:接口名限定方法名,并且没有public访问修饰符
8
9 {
10
11 Console.WriteLine("你好!");
12
13 }}
特别提醒:用于实现一个接口的所有方法都必须具有public可访问性。但是,假如使用显式接口实现,不应该为方法添加一个访问修饰符。
这时你只能这样调用:
IChineseGreeting otherChinese = new ChinesePeople();
otherChinese.SayHello();
不能通过类来调用
ChinesePeople chinese = new ChinesePeople();
chinese.SayHello(); //错误,不能访问。
记住:显示实现,只有接口可以访问。
显示接口实现看起来有点繁琐,而且还不能通过类来访问接口方法。为什么还要有“显式实现接口”呢?
其实,多个接口可能包含具有相同名称、相同返回类型和相同参数的方法。如果一个类实现了多个接口,而且接口中的方法具有相同的签名,就可以通过显式接口实现来消除它们之间的歧义。显式接口实现标识出哪个方法属于哪个接口。
IEnglishGreeting接口,如果需要具备英文的问好行为,需继承此接口。
可以发现,该接口中的SayHello方法签名和IChineseGreeting接口中一样。
1 class ChinesePeople:IChineseGreeting,IEnglishGreeting
2
3 {
4
5 void IChineseGreeting.SayHello()
6
7 {
8
9 Console.WriteLine("你好!");
10
11 }
12
13 void IEnglishGreeting.SayHello()
14
15 {
16
17 Console.WriteLine("Hello!");
18
19 }
20
21
22
23 }
可以这样调用:
ChinesePeople chinese = new ChinesePeople();
(chinese as IChineseGreeting).SayHello();
(chinese as IEnglishGreeting).SayHello();
----------------------------------------------------------------------------------------
此外,如果显示实现接口的类中还有该类特有的方法,此时会带来一些不便。
例如,使ChinesePeople具有Play()方法。
这就带来一个问题,要调用SayHello()方法时,必须通过接口实现,而此时Play()方法就无法访问。因为现在将对象看作IChineseGreeting接口了,而接口中没有Play()方法。
同样,如果通过类实现,就无法访问到SayHello()方法。
必须通过强制转换来解决这个问题。
IChineseGreeting chinese = new ChinesePeople();
chinese.SayHello();
//chinese.Play(); 错误,无法访问
(chinese as ChinesePeople).Play();//强制转换后,可以访问
隐式接口实现不存在此问题。
总结:
- 当类实现一个接口时,通常使用隐式接口实现,这样可以方便的访问接口方法和类自身具有的方法和属性。
- 当类实现多个接口时,并且接口中包含相同的方法签名,此时使用显式接口实现。即使没有相同的方法签名,仍推荐使用显式接口,因为可以标识出哪个方法属于哪个接口。
- 隐式接口实现,类和接口都可访问接口中方法。显式接口实现,只能通过接口访问。