第一种:编译时的多态性,直接这样说不知道说啥?程序执行过程主要分为三步:编译,链接,运行。在编译时体现的多态性是通过方法的重载来实现的。方法的重载我们听过,就是一个类对象调用若干同名,但参数类型、个数不同。类对象是根据参数类型,个数来决定调用哪个同名方法,实现何种操作。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); //显示派生类数据
} //两次调用完全相同的方法,实参不同,实现的功能不同
}
}
|
运行效果:
以上就是本文的全部内容,希望能够帮到大家。