1. 什么是多态
多态是C++中的一个重要的基础,面向对象编程语言中,接口的多种不同的实现方式即为多态。
2. 多态带来的好处
多态带来两个明显的好处:一是不用记大量的函数名了,二是它会依据调用时的上下文来确定实现。确定实现的过程由C++本身完成另外还有一个不明显但却很重要的好处是:带来了面向对象的编程。
3. C++中实现多态的方式
多态分为静态多态和动态多态。静态多态是通过重载和模板技术实现,在编译的时候确定。动态多态通过虚函数和继承关系来实现,执行动态绑定,在运行的时候确定。
C++中共有三种实现多态的方式。。第一种是函数重载;第二种是模板函数;第三种是虚函数(函数覆盖)。
4. 细说用函数重载实现的多态
函数重载是这样一种机制:允许有不同参数的函数有相同的名字。
具体一点讲就是:假如有如下三个函数:
void test(int arg){} //函数1
void test(char arg){} //函数2
void test(int arg1,int arg2){} //函数3
如果在C中编译,将会得到一个名字冲突的错误而不能编译通过。在C++中这样做是合法的。可是当我们调用test的时候到底是会调用上面三个函数中的哪一个呢?这要依据你在调用时给的出的参数来决定。如下:
test(5); //调用函数1
test('c');//调用函数2
test(4,5); //调用函数3
C++是如何做到这一点的呢?原来聪明的C++编译器在编译的时候悄悄的在我们的函数名上根据函数的参数的不同做了一些不同的记号。具体说如下:
void test(int arg) //被标记为 ‘test有一个int型参数’
void test(char arg) //被标记为 ‘test有一个char型的参数’
void test(int arg1,int arg2) //被标记为 ‘test第一个参数是int型,第二个参数为int型’
这样一来当我们进行对test的调用时,C++就可以根据调用时的参数来确定到底该用哪一个test函数了。噢,聪明的C++编译器。其实C++做标记做的比我上面所做的更聪明。我上面哪样的标记太长了。C++编译器用的标记要比我的短小的多。看看这个真正的C++的对这三个函数的标记:
?test@@YAXD@Z
?test@@YAXH@Z
?test@@YAXHH@Z
是不是短多了。但却不好看明白了。好在这是给计算机看的,人看不大明白是可以理解的。
还记得cout吧。我们用<<可以让它把任意类型的数据输出。比如可以象下面那样:
cout << 1; //输出int型
cout << 8.9; //输出double型
cout << 'a'; //输出char型
cout << "abc";//输出char数组型
cout << endl; //输出一个函数
cout之所以能够用一个函数名<<(<<是一个函数名)就能做到这些全是函数重载的功能。要是没有函数重载,我们也许会这样使用cout,如下:
cout int<< 1; //输出int型
cout double<< 8.9; //输出double型
cout char<< 'a'; //输出char型
cout charArray<< "abc"; //输出char数组型
cout function(…)<< endl; //输出函数
为每一种要输出的类型起一个函数名,这岂不是很麻烦呀。
不过函数重载有一个美中不足之处就是不能为返回值不同的函数进行重载。那是因为人们常常不为函数调用指出返回值。并不是技术上不能通过返回值来进行重载。
5. 细说用模板函数实现的多态
所谓模板函数(也有人叫函数模板)是这样一个概念:函数的内容有了,但函数的参数类型却是待定的(注意:参数个数不是待定的)。比如说一个(准确的说是一类或一群)函数带有两个参数,它的功能是返回其中的大值。这样的函数用模板函数来实现是适合不过的了。如下。
template < typename T>
T getMax(T arg1, T arg2)
{
return arg1 > arg2 ? arg1:arg2; //代码段1
}
这就是基于模板的多态吗?不是。因为现在我们不论是调用getMax(1, 2)还是调用getMax(3.0, 5.0)都是走的上面的函数定义。它没有根据调用时的上下文不同而执行不同的实现。所以这充其量也就是用了一个模板函数,和多态不沾边。怎样才能和多态沾上边呢?用模板特化呀!象这样:
template<>
char* getMax(char* arg1, char* arg2)
{
return (strcmp(arg1, arg2) > 0)?arg1:arg2;//代码段2
}
这样一来当我们调用getMax(“abc”, “efg”)的时候,就会执行代码段2,而不是代码段1。这样就是多态了。
更有意思的是如果我们再写这样一个函数:
char getMax(char arg1, char arg2)
{
return arg1>arg2?arg1:arg2; //代码段3
}
6虚函数实现多态详见http://www.cnblogs.com/wsw-seu/p/7646357.html