面向对象的特征封装、继承和多态。Polymorphism(多态性)来源于希腊单词,指“多种形态”。多态性的一个重要特征是方法的调用是在运行时确定而不是编译时。在.NET中用于实现多态性的关键词有virtual、override、abstract、interface。

virtual实现多态

shape类是通用的基类,draw是一个虚方法,每个派生类都可以有自己的override版本,在运行时可以用shape类的变量动态的调用draw方法。

public class Shape

{

public virtual void Draw()

{

Console.WriteLine("base class drawing");

}

}

public class Rectangle : Shape

{

public override void Draw()

{

Console.WriteLine("Drawing a Rectangle");

}

}

public class Square : Rectangle

{

public override void Draw()

{

Console.WriteLine("Drawing a Square");

base.Draw();

}

}

class Program

{

static void Main(string[] args)

{

System.Collections.Generic.List<Shape> shapes = new List<Shape>();

shapes.Add(new Rectangle());

shapes.Add(new Square());

foreach (Shape s in shapes)

{

s.Draw();

}

Console.ReadLine();

/*运行结果

Drawing a Rectangle

Drawing a Square

Drawing a Rectangle

*/

}

}

方法、属性、事件、索引器都可以被virtual修饰,但是字段不可以。派生类必须用override表示类成员参与虚调用。假如把Square中的draw方法替换为用new 修饰,则表示draw方法不参与虚调用,而且是一个新的方法,只是名字和基类方法重名。

public new void Draw()

{

Console.WriteLine("Drawing a Square");

base.Draw();

}

这个方法在Main方法中的foreach中将不会被调用,它不是虚方法了。用new修饰符后的程序运行结果,

/*运行结果

Drawing a Rectangle

Drawing a Rectangle

*/

假如说虚方法在rectangle扩展后,不希望square扩展了,可以在方法前加上sealed修饰符,

如下

public class Rectangle : Shape

{

public sealed override void Draw()

{

Console.WriteLine("Drawing a Rectangle");

}

}

当派生类重写某个虚拟成员时,即使该派生类的实例被当作基类的实例访问或者把派生类实例赋给父类变量进行访问,但是还是会调用派生类重写后的成员,可以把代码改为如下形式,

static void Main(string[] args)

{

System.Collections.Generic.List<Shape> shapes = new List<Shape>();

shapes.Add((Shape)new Rectangle());

shapes.Add((Shape)new Square());

foreach (Shape s in shapes)

{

s.Draw();

}

Console.ReadLine();

/*运行结果

Drawing a Rectangle

Drawing a Square

Drawing a Rectangle

*/

}

abstract实现多态

被abstract修饰的方法,默认是虚拟的,但是不能出现virtual关键词修饰。被abstract修饰的类可以有已实现的成员,可以有自己的字段,可以有非abstract 修饰的方法,但是不能实例化因为抽象的东西是没有实例对应的。比如,有人让我们画个图形(抽象)是画不出来的,但是让画个矩形(具体)是可以画出来的。下面是用abstract实现的多态版本,

public abstract class Shape

{

public abstract void Draw();

}

public class Rectangle : Shape

{

public  override void Draw()

{

Console.WriteLine("Drawing a Rectangle");

}

}

public class Square : Rectangle

{

public override void Draw()

{

Console.WriteLine("Drawing a Square");

base.Draw();

}

}

class Program

{

static void Main(string[] args)

{

System.Collections.Generic.List<Shape> shapes = new List<Shape>();

shapes.Add(new Rectangle());

shapes.Add(new Square());

foreach (Shape s in shapes)

{

s.Draw();

}

Console.ReadLine();

}

}

被abstract修饰的方法,在派生类中同样用override关键词进行扩展。同样可以用关键词sealed阻止派生类进行扩展。

interface实现多态

接口可由方法、属性、事件、索引器或这四种成员类型的任何组合构成。接口不能包含字段。接口成员默认是公共的,抽象的,虚拟的。若要实现接口成员,类中的对应成员必须是公共的、非静态的,并且与接口成员具有相同的名称和签名。下面是interface实现的多态版本

public interface IShape

{

void Draw();

}

public class Rectangle : IShape

{

public  void Draw()

{

Console.WriteLine("Drawing a Rectangle");

}

}

public class Square :  IShape

{

public  void Draw()

{

Console.WriteLine("Drawing a Square");

}

}

class Program

{

static void Main(string[] args)

{

System.Collections.Generic.List<IShape> shapes = new List<IShape>();

shapes.Add(new Rectangle());

shapes.Add(new Square());

foreach (IShape s in shapes)

{

s.Draw();

}

Console.ReadLine();

}

}

抽象类与接口

类可以实现无限个接口,但仅能从一个抽象(或任何其他类型)类继承。从抽象类派生的类仍可实现接口。msdn的在接口和抽象类的选择方面给的一些建议,

如果预计要创建组件的多个版本,则创建抽象类。抽象类提供简单易行的方法来控制组件版本。通过更新基类,所有继承类都随更改自动更新。另一方面,接口一旦创建就不能更改。如果需要接口的新版本,必须创建一个全新的接口。

如果创建的功能将在大范围的全异对象间使用,则使用接口。抽象类应主要用于关系密切的对象,而接口最适合为不相关的类提供通用功能。

如果要设计小而简练的功能块,则使用接口。如果要设计大的功能单元,则使用抽象类。

如果要在组件的所有实现间提供通用的已实现功能,则使用抽象类。抽象类允许部分实现类,而接口不包含任何成员的实现。

一个综合性的实例

public interface IShape

{

void Draw();

}

public class Shape:IShape

{

void IShape.Draw()

{

Console.WriteLine("Shape IShape.Draw()");

}

public virtual void Draw()

{

Console.WriteLine("Shape virtual Draw()");

}

}

public class Rectangle : Shape,IShape

{

void IShape.Draw()

{

Console.WriteLine("Rectangle IShape.Draw()");

}

public new virtual void Draw()

{

Console.WriteLine("Rectangle virtual Draw()");

}

}

public class Square : Rectangle

{

public override void Draw()

{

Console.WriteLine("Square override Draw()");

}

}

class Program

{

static void Main(string[] args)

{

Square squre = new Square();

Rectangle rect = squre;

Shape shape = squre;

IShape ishape = squre;

squre.Draw();

rect.Draw();

shape.Draw();

ishape.Draw();

Console.ReadLine();

}

}

/*运行结果:

Square override Draw()①

Square override Draw()②

Shape virtual Draw()③

Rectangle IShape.Draw()④

*/

在这个程序里,把派生类实例赋给父类变量或者接口。对结果①无需解释。结果②,因为Draw方法是虚方法,虚方法的调用规则是调用离实例变量最近的override版本方法,Square类中的Draw方法是离实例square最近的方法,即使是把Square类型的实例赋值给Rectangle类型的变量去访问,仍然调用的是Square类重写的方法。对于结果③,也是虚方法调用,在子类Rectangle中的draw方法用new修饰,这就表明shape类中的virtual到此中断,后面Square中的override版是针对Rectangle中的Draw方法,此时,离square实例最近的实现就是Shape类中的Draw 方法,因为Shape类中的Draw方法没有override的版本只能调用本身的virtual版了。结果④,因为Rectangle重新声明实现接口IShape,接口调用同样符合虚方法调用规则,调用离它最近的实现,Rectangle中的实现比Shape中的实现离实例square更近。Rectangle中的IShape.Draw()方法是显式接口方法实现,对于它不能有任何的访问修饰符,只能通过接口变量访问它,同时也不能用virtual或者override进行修饰,也不能被派生类型调用。只能用IShape变量进行访问。如果类型中有显式接口的实现,而且用的是接口变量,默认调用显式接口的实现方法。

override和方法选择

public class Base

{

public virtual void Write(int num)

{

Console.WriteLine("int:" + num.ToString());

}

}

public class Derived : Base

{

public override void Write(int num)

{

Console.WriteLine("derived:" + num.ToString());

}

public void Write(double num)

{

Console.WriteLine("derived double:" + num.ToString());

}

}

class Program

{

static void Main(string[] args)

{

int val = 5;

Derived d = new Derived();

d.Write(val);

Console.ReadLine();

}

}

/*运行结果:

derived double:5

*/

在 Derived 的一个实例中调用 Write时,C# 编译器将首先尝试使该调用与最初在 Derived 上声明的 Write版本兼容。重写方法不被视为是在类上进行声明的,而是在基类上声明的方法的新实现。由于变量val可以隐式的转换为double型的,所以C#编译器选择Write(double num)。仅当 C# 编译器无法将方法调用与 Derived 上的原始方法匹配时,它才尝试将该调用与具有相同名称和兼容参数的重写方法匹配。

C#中的多态现象的更多相关文章

  1. JavaSE自学笔记

    ch03 [Thu Aug 18 2016 11:22:26 GMT+0800] 对象变量与对象之间是指代关系,对象变量并不能完全说明有无对象可用.这种指代关系是通过赋值运算建立起来的.对象变量保存的 ...

  2. Python开源框架

    info:更多Django信息url:https://www.oschina.net/p/djangodetail: Django 是 Python 编程语言驱动的一个开源模型-视图-控制器(MVC) ...

  3. 深入Java核心 Java中多态的实现机制(1)

    在疯狂java中,多态是这样解释的: 多态:相同类型的变量,调用同一个方法时,呈现出多中不同的行为特征, 这就是多态. 加上下面的解释:(多态四小类:强制的,重载的,参数的和包含的) 同时, 还用人这 ...

  4. java中abstract

    abstract(抽象)修饰符,可以修饰类和方法 1,abstract修饰类,会使这个类成为一个抽象类,这个类将不能生成对象实例,但可以做为对象变量声明的类型,也就是编译时类型,抽象类就像当于一类的半 ...

  5. java中实现多态的机制是什么?

    多态性是面向对象程序设计代码重用的一个重要机制,我们曾不只一次的提到Java多态性.在Java运行时多态性:继承和接口的实现一文中,我们曾详细介绍了Java实现运行时多态性的动态方法调度:今天我们再次 ...

  6. C++中多态性学习(上)

    多态性学习(上) 什么是多态? 多态是指同样的消息被不同类型的对象接收时导致不同的行为.所谓消息是指对类的成员函数的调用,不同的行为是指不同的实现,也就是调用了不同的函数.虽然这看上去好像很高级的样子 ...

  7. (C/C++学习)5.C++中的虚继承-虚函数-多态解析

    说明:在C++学习的过程中,虚继承-虚函数经常是初学者容易产生误解的两个概念,它们与C++中多态形成的关系,也是很多初学者经常产生困惑的地方,这篇文章将依次分别对三者进行解析,并讲述其之间的联系与不同 ...

  8. C++中的纯虚函数

    ---恢复内容开始--- 在C++中的一种函数申明被称之为:纯虚函数(pure virtual function).它的申明格式如下 class CShape { public: ; }; 在什么情况 ...

  9. java中abstract怎么使用

    abstract(抽象)修饰符,可以修饰类和方法 1,abstract修饰类,会使这个类成为一个抽象类,这个类将不能生成对象实例,但可以做为对象变量声明的类型,也就是编译时类型,抽象类就像当于一类的半 ...

随机推荐

  1. 调试Python代码的工具

    pdb: 首先来说Python里内建的调试器,pdb.它利用一个简单的命令行界面,还有很多你在用调试器时用得上的功能.帮助系统能为你指出你能运行的命令,比如单步调试代码,操纵调用栈和设置断点. 一些它 ...

  2. cocos2d-x 3.2 DrawNode 绘图API

    关于Cocos2d-x 3.x 版本的绘图方法有两种: 1.使用DrawNode类绘制自定义图形. 2.继承Layer类重写draw()方法. 以上两种方法都可以绘制自定义图形,根据自己的需要选择合适 ...

  3. 【多线程】Java并发编程:并发容器之CopyOnWriteArrayList(转载)

    原文链接: http://ifeve.com/java-copy-on-write/ Copy-On-Write简称COW,是一种用于程序设计中的优化策略.其基本思路是,从一开始大家都在共享同一个内容 ...

  4. 创建一个EMS 扩展包

    EMS Package 向导: File > New > Other > Delphi projects > EMS > EMS Package Empty packag ...

  5. 红米手机拍照效果测评(对比小米2A)

    小米相关的产品一向都很很受用户的欢迎,一个就是实惠,另一个就是配置还不错.近期小米推出的红米手机可谓是先声夺人,关注度异常火爆.今天刚抢的红米快递寄到了,来测试下红米手机的拍照表现,800万像素怎么样 ...

  6. Codeforces 543D. Road Improvement (树dp + 乘法逆元)

    题目链接:http://codeforces.com/contest/543/problem/D 给你一棵树,初始所有的边都是坏的,要你修复若干边.指定一个root,所有的点到root最多只有一个坏边 ...

  7. UVA 11983 Weird Advertisement(线段树求矩形并的面积)

    UVA 11983 题目大意是说给你N个矩形,让你求被覆盖k次以上的点的总个数(x,y<1e9) 首先这个题有一个转化,吧每个矩形的x2,y2+1这样就转化为了求N个矩形被覆盖k次以上的区域的面 ...

  8. lib和dll的例子

    .dll和.lib的区别 lib是静态库,dll一般是动态链接库(也有可能是别的)比如要编译个exe,lib在编译的时候就会被编译到exe里,作为程序的一部分而dll是不被编译进去,是运行的时候才调入 ...

  9. [OAuth2 & OpenID] 1.OAuth2授权

    1 OAuth2解决什么问题的? 举个栗子先.小明在QQ空间积攒了多年的照片,想挑选一些照片来打印出来.然后小明在找到一家提供在线打印并且包邮的网站(我们叫它PP吧(Print Photo缩写

  10. ios和android一并学习的体会

    如果说为什么要同时学习这两种不同的移动平台,其实有一定的“闲”的因素在里面. 相对于ios,android我是早半年接触的.最开始学习的时候也就是j2ee学习的延续,通过看视频连带看书学了大概一个月的 ...