假设我们有两个具有冲突方法签名的接口:
interface IA
{
void F();
}
interface IB
{
int F();
}
现在我们创建一个这样的类:
class Test : IA, IB
{
public void F() { ... }
int IB.F() { ... }
}
名为IB.F
的方法看起来像private
方法,但您可以执行以下操作:
var t = new Test();
t.F(); //Calls public method F
((IB)t).F(); //Calls private method IB.F!!!
所以我的问题是: C#编译器如何知道可以在类范围外调用IB.F
?
我猜这是IB.
前缀,但查看了IL
代码,它只显示为带有奇怪签名的private
方法。
答案 0 :(得分:9)
“private”方法包含以下内容:
.method private hidebysig newslot virtual final instance int32 ns.IB.F() cil managed
{
.override Tests.IB::F
//...
}
在IB
:
.method public hidebysig newslot abstract virtual instance int32 F() cil managed
{
}
.override
告诉编译器查看IB
中的声明。它说方法是public
。
该方法实际上没有声明private
,默认情况下,“nothing”对于接口方法来说意味着public
。
尝试在int IB.F()
前添加任何类型的访问修饰符,这将是编译器错误,因为您可能无法更改显式接口实现的可访问性。
答案 1 :(得分:2)
看起来像
的方法int IB.F() { ... }
即。名称中有一个点.
,并且(最后一个)点之前的名称部分是接口名称,不是普通的实例方法。这是一个所谓的 显式接口实现 。
不要将其视为通常的private
方法。请注意,将C#关键字private
与显式接口实现一起使用甚至是不合法的。
从某种意义上说,这种方法具有它实现的接口的可访问性。
因此,只要有人拥有该类的实例,就可以调用它来转换为接口。
例如,在MSDN网站上,他们不会记录班级的internal
或private
成员。但是他们做文档显式接口实现(接口在程序集外部可见),因为这些成员可以从外部调用(即由我们调用)。有关示例,请参阅List<T>
Class。