| 添加到收藏夹 | 返回目录页 | 上一篇:基于C#的接口基础教程之三 |
基于C#的接口基础教程之四
第四节、访问接口 对接口成员的访问 对接口方法的调用和采用索引指示器访问的规则与类中的情况也是相同的。如果底层成员的命名与继承而来的高层成员一致,那么底层成员将覆盖同名的高层成员。但由于接口支持多继承,在多继承中,如果两个父接口含有同名的成员,这就产生了二义性(这也正是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#的接口,请看下一节-实现接口
- · 基于C#的接口基础教程之五
- · 基于C#的接口基础教程之六
- · 基于C#的接口基础教程之七
- · 使用ADO.NET设计数据库应用程序
- · 创建基于ASP.NET的SMTP邮件服务
- · ASP.NET中编程杀死进程
- · WebMatrix开发ASP.NET试用手记
- · 教你如何把一篇文章按要求分段
- · 在HTML页面中实现点击数统计
- · ASP字数计算函数
- · 创建ASP.NET监视服务器进程
- · 使用更精简的代码保证ASP.NET应用程序的安全
- · 创建用于ASP.NET的分页程序控件
- · jsp+javascript打造级连菜单
- · 为ASP.NET应用缓存Oracle数据
- · 用C#开发网络防火墙技术分析
- · C#打造自己的文件浏览器
- · ASP.NET结合COM组件发送Email
- · ASP.NET HTTP运行时组成详解
- · ASPX页Web服务调用性能优化
- · 从 PHP 迁移到 ASP.NET
- · SQL Server中全角和半角字符的比较问题
- · 在C#中利用SharpZipLib进行文件的压缩和解压缩
- · C#就是Java只不过差了一点点
- · 无刷新随时取得用户当前活动信息
- · 教学体会: ADO.NET的连接式和断开式
- · JSP与XML的结合
- · 在VB中利用Word宏命令开发ASP组件
- · ASP学习:urldecode 方法补遗
- · ASP.NET中Datagrid常见错误
- · ASP之对象总结
- · 用ASP.NET和XML做的新闻系统
- · ASP.NET保持用户状态的九种选择下
- · ASP 编程中20个非常有用的例子
- · 数据库的查询优化技术
- · ASP.NET中Cookie编程的基础知识
- · ASP.NET中Cookie编程简明参考
- · 在状态栏中实现活动文字效果
- · 在Zeus Web Server中安装PHP语言支持
- · 在PHP中使用XML
- · 使用w3Sockets组件实现域名查询功能
- · 用SQL 2000创建用户化XML流
- · 基于Nokia S60的游戏开发之四
- · 将代码页从SQL Server 7.0改变到SQL Server 2000
- · 用VS2003调试ASP的方法和体会
- · 将一个更新划分为几个批次
- · ASP.NET中利用cookies保持客户端信息
- · J2ME编程之Nokia 7210配置篇
- · SQL Server数据库性能的优化
- · 一次编写,随处运行
- · 用XMLHTTP做一个自己特色的Google
- · 建立JSP操作以提高数据库访问的效率
- · PHPShop存在多个安全漏洞
- · JavaBean实现多文件上传的两种方法
- · asp.net 实现“九连环”小游戏
- · 概述IE和SQL2k开发一个XML聊天程序
- · 学习使用ASP对象和组件
- · ADO.net中数据库连接方式
- · ASP中存储过程调用的两种方式及比较
- · javascript中的数组应用的一点发现

