第一种:编译时的多态性,直接这样说不知道说啥?程序执行过程主要分为三步:编译,链接,运行。在编译时体现的多态性是通过方法的重载来实现的。方法的重载我们听过,就是一个类对象调用若干同名,但参数类型、个数不同。类对象是根据参数类型,个数来决定调用哪个同名方法,实现何种操作。c#语言的方法重载与c++、java基本是一样的,但是c#还包含c++的运算符重载,java里面没有这个概念。
第二种:运行时的多态性,是在系统运行时,调用一个名字相同,参数类型也相同以及个数也完全一样的方法,但会完成不同的操作。c#运行时的多态性通过虚方法实现。在类方法声明加上了virtual修饰符,称为虚方法,反之为非虚方法。就这一点c#跟c++有很多相似之处。
代码清单:虚方法与非虚方法的区别
| 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 |
namespace e_11
{
class a
{
public void f() //非虚方法
{ console.writeline("a.f"); }
public virtual void g() //虚方法
{ console.writeline("a.g"); }
}
class b : a //a类为b类的基类
{
new public void f() //隐藏基类的同名非虚方法f(),注意使用new
{ console.writeline("b.f"); }
public override void g() //隐藏基类的同名虚方法g(),注意使用override
{ console.writeline("b.g"); }
}
class test
{
static void f2(a aa) //注意,参数为a类引用变量
{ aa.g(); }
static void main(string[] args)
{
b b = new b();
a a1 = new a();
a a2 = b; //允许基类引用变量引用派生类对象,a2引用a类的派生类b的对象b
a1.f(); //调用基类a的非虚方法f(),显示a.f
a2.f(); //f()为非虚方法,调用基类a的f(),显示a.f
b.f(); //f()为非虚方法,调用派生类的f(),显示b.f
a1.g(); //g()为虚方法,因a1引用基类a对象,调用基类a的g(),显示a.g
a2.g(); //g()为虚方法,因a2引用派生类b对象,调用派生类b的g(),显示b.g
f2(a2); //由于a2引用a类派生类b的对象b,调用派生类b的的g(),显示b.g
f2(a1); //实参为基类a对象,调用a类的函数g(),显示a.g
}
}
}
|
输出结果:

看上面的代码,我产生了一个疑问?为什么因为g()为虚方法,a2引用派生类b对象,调用的是b类的g()?
这里有一个解释:如果将基类引用变量引用不同对象,或者是基类对象,或者是派生类对象,用这个基类引用变量分别调用同名虚方法,根据对象的不同,会完成不同的操作。而非虚方法不具备次此功能。
关于类的多态性典型的例子:
| 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 |
//类的多态性
namespace e_11
{
public class person
{
private string name = "张三"; //类的成员说明
private int age = 12;
protected virtual void display() //类的虚方法
{ console.writeline("姓名:{0},年龄:{1}", name, age); }
public person(string name, int age) //构造函数,函数名和类名同名,无返回值
{ name = name; age = age; }
static public void displaydata(person aperson) //静态方法
{ aperson.display(); } //不是静态方法调用实例方法,如写为display()错误
}
public class employe : person //基类是person
{
private string department;
private decimal salary;
public employe(string name, int age, string d, decimal s)
: base(name, age)
{ department = d; salary = s; }
protected override void display() //重载虚方法,注意用override
{
base.display();
console.writeline("部门:{0}, 薪金:{1}", department, salary);
}
}
class class1
{
static void main(string[] args)
{
person oneperson = new person("李四", 30);
person.displaydata(oneperson); //显示基类的数据
employe oneemployee = new employe("王五", 40, "财务", 2000);
person.displaydata(oneemployee); //显示派生类数据
}//两次调用完全相同的方法,实参不同,实现的功能不同
}
}
|
运行效果:

以上就是本文的全部内容,希望能够帮到大家。








发表评论
◎欢迎参与讨论,请在这里发表您的看法、交流您的观点。