[转载]虚函数与默认实参
本文转载于http://blog.csdn.net/chenyiming_1990/article/details/8981868
新加内容提醒:
关于利用virtual函数实现动态绑定/多态,需要尤其注意的一点是:
必须保证基类和派生类中对应的虚函数除了函数名称一致之外,还必须保证形参和返回值也完全一致!
若基类和派生类中仅仅是虚函数的函数名称一致,但是形参不一致,则无法实现动态绑定/多态!实际上造成的结果是:派生类中对基类中对应的函数进行了隐藏,在派生类中只能访问派生类中的该函数,在基类中只能访问基类中对应的同名函数。
针对基类和派生类中的虚函数有没有默认实参问题进行讨论。
1.基类虚函数不带参数,派生类带参数
#include<iostream> using namespace std; class super { public: virtual void somemethod()const { cout<<"base" << endl; } }; class sub : public super { public: virtual void somemethod(int i = 12)const { cout<<"sub" << "i = " << i <<endl; } }; int main(void) { sub mysub; super &ref = mysub; ref.somemethod(); // 调用哪个? return 0; }
输出:base
调用的是基类的,派生类是基类这个函数的重新定义的版本,无法实现动态绑定/多态。(这个务必引起重视,因为形参已经不一样了)
2.基类虚函数不带参数,派生类带参数
[cpp] view plaincopyprint? #include<iostream> using namespace std; class super { public: virtual void somemethod(int i = 21)const {cout<<"base"<<"i = "<< i <<endl;} }; class sub : public super { public: virtual void somemethod()const { cout<<"sub" << "i = " << i <<endl; } }; int main(void) { sub mysub; super &ref = mysub; ref.somemethod(); // 调用哪个? return 0; }
输出:
Base i = 21
原因和上面的一样。调用的是基类的,派生类是基类这个函数的重新定义的版本,无法实现动态绑定/多态。(这个务必引起重视,因为形参已经不一样了)
3.基类虚函数带参数,派生类带参数,且默认实参是一样的。
#include<iostream> using namespace std; class super { public: virtual void somemethod(int i = 12)const {cout<<"base "<<"i = "<< i <<endl;} }; class sub : public super { public: virtual void somemethod(int i = 12)const {cout<<"sub "<<"i = "<< i <<endl;} } int main(void) { sub mysub; super &ref = mysub; ref.somemethod(); // 调用哪个? return 0; }
输出:sub i = 12
4.基类虚函数带参数,派生类带参数,且默认实参不一样的。
#include<iostream> using namespace std; class super { public: virtual void somemethod(int i = 12)const {cout<<"base "<<"i = "<< i <<endl;} }; class sub : public super { public: virtual void somemethod(int i = 21)const { cout<<"sub " << "i = " << i <<endl;} }; int main(void) { sub mysub; super &ref = mysub; ref.somemethod(); // 调用哪个? return 0; }
输出:sub i = 12
原因:你会发现虽然调用的是派生类的函数,但是实参确实基类的。这就是比较变态的地方。
天行健,君子以自强不息;地势坤,君子以厚德载物。