分享
 
 
 

VisualC#访问接口

王朝c#·作者佚名  2008-05-19
窄屏简体版  字體: |||超大  

对接口成员的访问

对接口方法的调用和采用索引指示器访问的规则与类中的情况也是相同的。如果底层成员的命名与继承而来的高层成员一致,那么底层成员将覆盖同名的高层成员。但由于接口支持多继承,在多继承中,如果两个父接口含有同名的成员,这就产生了二义性(这也正是C#中取消了类的多继承机制的原因之一),这时需要进行显式的定义:

using System ;

interface ISequence {

int Count { get; set; }

}

interface IRing {

void Count(int i) ;

}

interface IRingSequence: ISequence, IRing { }

class CTest {

void Test(IRingSequence rs) {

//rs.Count(1) ; 错误, Count 有二义性

//rs.Count = 1; 错误, Count 有二义性

((ISequence)rs).Count = 1; // 正确

((IRing)rs).Count(1) ; // 正确调用IRing.Count

}

}

上面的例子中,前两条语句rs .Count(1)和rs .Count = 1会产生二义性,从而导致编译时错误,因此必须显式地给rs 指派父接口类型,这种指派在运行时不会带来额外的开销。

再看下面的例子:

using System ;

interface IInteger {

void Add(int i) ;

}

interface IDouble {

void Add(double d) ;

}

interface INumber: IInteger, IDouble {}

class CMyTest {

void Test(INumber Num) {

// Num.Add(1) ; 错误

Num.Add(1.0) ; // 正确

((IInteger)n).Add(1) ; // 正确

((IDouble)n).Add(1) ; // 正确

}

}

调用Num.Add(1) 会导致二义性,因为候选的重载方法的参数类型均适用。但是,调用Num.Add(1.0) 是允许的,因为1.0 是浮点数参数类型与方法IInteger.Add()的参数类型不一致,这时只有IDouble.Add 才是适用的。不过只要加入了显式的指派,就决不会产生二义性。

接口的多重继承的问题也会带来成员访问上的问题。例如:

interface IBase {

void FWay(int i) ;

}

interface ILeft: IBase {

new void FWay (int i) ;

}

interface IRight: IBase

{ void G( ) ; }

interface IDerived: ILeft, IRight { }

class CTest {

void Test(IDerived d) {

d. FWay (1) ; // 调用ILeft. FWay

((IBase)d). FWay (1) ; // 调用IBase. FWay

((ILeft)d). FWay (1) ; // 调用ILeft. FWay

((IRight)d). FWay (1) ; // 调用IBase. FWay

}

}

上例中,方法IBase.FWay在派生的接口ILeft中被Ileft的成员方法FWay覆盖了。所以对d. FWay (1)的调用实际上调用了。虽然从IBase-> IRight-> IDerived这条继承路径上来看,ILeft.FWay方法是没有被覆盖的。我们只要记住这一点:一旦成员被覆盖以后,所有对其的访问都被覆盖以后的成员"拦截"了。

类对接口的实现

前面我们已经说过,接口定义不包括方法的实现部分。接口可以通过类或结构来实现。我们主要讲述通过类来实现接口。用类来实现接口时,接口的名称必须包含在类定义中的基类列表中。

下面的例子给出了由类来实现接口的例子。其中ISequence 为一个队列接口,提供了向队列尾部添加对象的成员方法Add( ),IRing 为一个循环表接口,提供了向环中插入对象的方法Insert(object obj),方法返回插入的位置。类RingSquence 实现了接口ISequence 和接口IRing。

using System ;

interface ISequence {

object Add( ) ;

}

interface ISequence {

object Add( ) ;

}

interface IRing {

int Insert(object obj) ;

}

class RingSequence: ISequence, IRing

{

public object Add( ) {…}

public int Insert(object obj) {…}

}

如果类实现了某个接口,类也隐式地继承了该接口的所有父接口,不管这些父接口有没有在类定义的基类表中列出。看下面的例子:

using System ;

interface IControl {

void Paint( );

}

interface ITextBox: IControl {

void SetText(string text);

}

interface IListBox: IControl {

void SetItems(string[] items);

}

interface IComboBox: ITextBox, IListBox { }

这里, 接口IcomboBox继承了ItextBox和IlistBox。类TextBox不仅实现了接口ITextBox,还实现了接口ITextBox 的父接口IControl。

前面我们已经看到,一个类可以实现多个接口。再看下面的例子:

interface IDataBound {

void Bind(Binder b);

}

public class EditBox: Control, IControl, IDataBound {

public void Paint( );

public void Bind(Binder b) {...}

}

类EditBox从类Control中派生并且实现了Icontrol和IdataBound。在前面的例子中接口Icontrol中的Paint方法和IdataBound接口中的Bind方法都用类EditBox中的公共成员实现。C#提供一种实现这些方法的可选择的途径,这样可以使执行这些的类避免把这些成员设定为公共的。接口成员可以用有效的名称来实现。例如,类EditBox可以改作方法Icontrol.Paint和IdataBound.Bind来来实现。

public class EditBox: IControl, IDataBound {

void IControl.Paint( ) {...}

void IDataBound.Bind(Binder b) {...}

}

因为通过外部指派接口成员实现了每个成员,所以用这种方法实现的成员称为外部接口成员。外部接口成员可以只是通过接口来调用。例如,Paint方法中EditBox的实现可以只是通过创建Icontrol接口来调用。

class Test {

static void Main( ) {

EditBox editbox = new EditBox( );

editbox.Paint( ); //错误: EditBox 没有Paint 事件

IControl control = editbox;

control.Paint( ); // 调用 EditBox的Paint事件

}

}

上例中,类EditBox 从Control 类继承并同时实现了IControl and IDataBound 接口。EditBox 中的Paint 方法来自IControl 接口,Bind 方法来自IDataBound 接口,二者在EditBox 类中都作为公有成员实现。当然,在C# 中我们也可以选择不作为公有成员实现接口。

如果每个成员都明显地指出了被实现的接口,通过这种途径被实现的接口我们称之为显式接口成员(explicit interface member)。 用这种方式我们改写上面的例子:

public class EditBox: IControl, IDataBound {

void IControl.Paint( ) {…}

void IDataBound.Bind(Binder b) {…}

}

显式接口成员只能通过接口调用。例如:

class CTest {

static void Main( ) {

EditBox editbox = new EditBox( ) ;

editbox.Paint( ) ; //错误:不同的方法

IControl control = editbox;

control.Paint( ) ; //调用 EditBox的Paint方法

}

}

上述代码中对editbox.Paint( )的调用是错误的,因为editbox 本身并没有提供这一方法。control.Paint( )是正确的调用方式。

注释:接口本身不提供所定义的成员的实现,它仅仅说明这些成员,这些成员必须依靠实现接口的类或其它接口的支持。

知道了怎样访问接口,我们还要知道怎样实现接口,要实现C#的接口,请看下一节-实现接口

类对接口的实现

前面我们已经说过,接口定义不包括方法的实现部分。接口可以通过类或结构来实现。我们主要讲述通过类来实现接口。用类来实现接口时,接口的名称必须包含在类定义中的基类列表中。

下面的例子给出了由类来实现接口的例子。其中ISequence 为一个队列接口,提供了向队列尾部添加对象的成员方法Add( ),IRing 为一个循环表接口,提供了向环中插入对象的方法Insert(object obj),方法返回插入的位置。类RingSquence 实现了接口ISequence 和接口IRing。

using System ;

interface ISequence {

object Add( ) ;

}

interface ISequence {

object Add( ) ;

}

interface IRing {

int Insert(object obj) ;

}

class RingSequence: ISequence, IRing

{

public object Add( ) {…}

public int Insert(object obj) {…}

}

如果类实现了某个接口,类也隐式地继承了该接口的所有父接口,不管这些父接口有没有在类定义的基类表中列出。看下面的例子:

using System ;

interface IControl {

void Paint( );

}

interface ITextBox: IControl {

void SetText(string text);

}

interface IListBox: IControl {

void SetItems(string[] items);

}

interface IComboBox: ITextBox, IListBox { }

这里, 接口IcomboBox继承了ItextBox和IlistBox。类TextBox不仅实现了接口ITextBox,还实现了接口ITextBox 的父接口IControl。

前面我们已经看到,一个类可以实现多个接口。再看下面的例子:

interface IDataBound {

void Bind(Binder b);

}

public class EditBox: Control, IControl, IDataBound {

public void Paint( );

public void Bind(Binder b) {...}

}

类EditBox从类Control中派生并且实现了Icontrol和IdataBound。在前面的例子中接口Icontrol中的Paint方法和IdataBound接口中的Bind方法都用类EditBox中的公共成员实现。C#提供一种实现这些方法的可选择的途径,这样可以使执行这些的类避免把这些成员设定为公共的。接口成员可以用有效的名称来实现。例如,类EditBox可以改作方法Icontrol.Paint和IdataBound.Bind来来实现。

public class EditBox: IControl, IDataBound {

void IControl.Paint( ) {...}

void IDataBound.Bind(Binder b) {...}

}

因为通过外部指派接口成员实现了每个成员,所以用这种方法实现的成员称为外部接口成员。外部接口成员可以只是通过接口来调用。例如,Paint方法中EditBox的实现可以只是通过创建Icontrol接口来调用。

class Test {

static void Main( ) {

EditBox editbox = new EditBox( );

editbox.Paint( ); //错误: EditBox 没有Paint 事件

IControl control = editbox;

control.Paint( ); // 调用 EditBox的Paint事件

}

}

上例中,类EditBox 从Control 类继承并同时实现了IControl and IDataBound 接口。EditBox 中的Paint 方法来自IControl 接口,Bind 方法来自IDataBound 接口,二者在EditBox 类中都作为公有成员实现。当然,在C# 中我们也可以选择不作为公有成员实现接口。

如果每个成员都明显地指出了被实现的接口,通过这种途径被实现的接口我们称之为显式接口成员(explicit interface member)。 用这种方式我们改写上面的例子:

public class EditBox: IControl, IDataBound {

void IControl.Paint( ) {…}

void IDataBound.Bind(Binder b) {…}

}

显式接口成员只能通过接口调用。例如:

class CTest {

static void Main( ) {

EditBox editbox = new EditBox( ) ;

editbox.Paint( ) ; //错误:不同的方法

IControl control = editbox;

control.Paint( ) ; //调用 EditBox的Paint方法

}

}

上述代码中对editbox.Paint( )的调用是错误的,因为editbox 本身并没有提供这一方法。control.Paint( )是正确的调用方式。

注释:接口本身不提供所定义的成员的实现,它仅仅说明这些成员,这些成员必须依靠实现接口的类或其它接口的支持。

 
 
 
免责声明:本文为网络用户发布,其观点仅代表作者个人观点,与本站无关,本站仅提供信息存储服务。文中陈述内容未经本站证实,其真实性、完整性、及时性本站不作任何保证或承诺,请读者仅作参考,并请自行核实相关内容。
2023年上半年GDP全球前十五强
 百态   2023-10-24
美众议院议长启动对拜登的弹劾调查
 百态   2023-09-13
上海、济南、武汉等多地出现不明坠落物
 探索   2023-09-06
印度或要将国名改为“巴拉特”
 百态   2023-09-06
男子为女友送行,买票不登机被捕
 百态   2023-08-20
手机地震预警功能怎么开?
 干货   2023-08-06
女子4年卖2套房花700多万做美容:不但没变美脸,面部还出现变形
 百态   2023-08-04
住户一楼被水淹 还冲来8头猪
 百态   2023-07-31
女子体内爬出大量瓜子状活虫
 百态   2023-07-25
地球连续35年收到神秘规律性信号,网友:不要回答!
 探索   2023-07-21
全球镓价格本周大涨27%
 探索   2023-07-09
钱都流向了那些不缺钱的人,苦都留给了能吃苦的人
 探索   2023-07-02
倩女手游刀客魅者强控制(强混乱强眩晕强睡眠)和对应控制抗性的关系
 百态   2020-08-20
美国5月9日最新疫情:美国确诊人数突破131万
 百态   2020-05-09
荷兰政府宣布将集体辞职
 干货   2020-04-30
倩女幽魂手游师徒任务情义春秋猜成语答案逍遥观:鹏程万里
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案神机营:射石饮羽
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案昆仑山:拔刀相助
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案天工阁:鬼斧神工
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案丝路古道:单枪匹马
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:与虎谋皮
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:李代桃僵
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:指鹿为马
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案金陵:小鸟依人
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案金陵:千金买邻
 干货   2019-11-12
 
推荐阅读
 
 
 
>>返回首頁<<
 
靜靜地坐在廢墟上,四周的荒凉一望無際,忽然覺得,淒涼也很美
© 2005- 王朝網路 版權所有