C++的重载(overload)与重写(override)
成员函数被重载的特征:
(1)相同的范围(在同一个类中);
(2)函数名字相同;
(3)参数不同;
(4)virtual关键字可有可无。
重写是指派生类函数重写基类函数,是C++的多态的表现,特征是:
(1)不同的范围(分别位于派生类与基类);
(2)函数名字相同;
(3)参数相同;
(4)基类函数必须有virtual关键字。
示例中,函数Base::f(int)与Base::f(float)相互重载,而Base::g(void)被Derived::g(void)重写。
1 #include <iostream>
2 using namespace std;
3
4 class Base
5 {
6 public:
7 void f(int x){ cout << "Base::f(int) " << x << endl; }
8 void f(float x){ cout << "Base::f(float) " << x << endl; }
9 virtual void g(void){ cout << "Base::g(void)" << endl;}
10 };
11
12 class Derived : public Base
13 {
14 public:
15 virtual void g(void){ cout << "Derived::g(void)" << endl;}
16 };
17
18 int main()
19 {
20 Derived d;
21 Base *pb = &d;
22 pb->f(42); // Base::f(int) 42
23 pb->f(3.14f); // Base::f(float) 3.14
24 pb->g(); // Derived::g(void)
25
26 return 0;
27 }
令人迷惑的隐藏规则
本来仅仅区别重载与重写并不算困难,但是C++的隐藏规则(遮蔽现象)使问题复杂性陡然增加。这里“隐藏”是指派生类的函数屏蔽了与其同名的基类函数,规则如下:
(1)如果派生类的函数与基类的函数同名,但是参数不同。此时,不论有无virtual关键字,基类的函数将被隐藏。
(2)如果派生类的函数与基类的函数同名,并且参数也相同,但是基类函数没有virtual关键字。此时,基类的函数被隐藏。
这种隐藏规则,不仅仅是表现在对成员函数上,对同名的data member也是如此。
示例程序中:
(1)函数Derived::f(float)重写了Base::f(float)。
(2)函数Derived::g(int)隐藏了Base::g(float)。
(3)函数Derived::h(float)隐藏了Base::h(float)。
1 #include <iostream>
2 using namespace std;
3
4 class Base
5 {
6 public:
7 virtual void f(float x){ cout << "Base::f(float) " << x << endl; }
8 virtual void g(float x){ cout << "Base::g(float) " << x << endl; }
9 void h(float x){ cout << "Base::h(float) " << x << endl; }
10 };
11
12 class Derived : public Base
13 {
14 public:
15 virtual void f(float x){ cout << "Derived::f(float) " << x << endl; }
16 virtual void g(int x){ cout << "Derived::g(int) " << x << endl; }
17 void h(float x){ cout << "Derived::h(float) " << x << endl; }
18 };
19
20 int main()
21 {
22 Derived d;
23 Base *pb = &d;
24 Derived *pd = &d;
25
26 // Good : behavior depends solely on type of the object
27 pb->f(3.14f); // Derived::f(float) 3.14
28 pd->f(3.14f); // Derived::f(float) 3.14
29
30 // Bad : behavior depends on type of the pointer
31 pb->g(3.14f); // Base::g(float) 3.14 (surprise!)
32 pd->g(3.14f); // Derived::g(int) 3
33
34 // Bad : behavior depends on type of the pointer
35 pb->h(3.14f); // Base::h(float) 3.14 (surprise!)
36 pd->h(3.14f); // Derived::h(float) 3.14
37
38 return 0;
39 }
另一个关于虚函数很微妙的错误情况:参数相同,但是基类的函数是const的,派生类的函数却不是。
1 #include <iostream>
2 using namespace std;
3
4 class Base
5 {
6 public:
7 virtual void f(float x){ cout << "Base::f(float) " << x << endl; }
8 };
9
10 class Derived : public Base
11 {
12 public:
13 virtual void f(float x) const { cout << "Derived::f(float) " << x << endl; }
14 };
15
16 int main()
17 {
18 Derived d;
19 Base *pb = &d;
20 Derived *pd = &d;
21
22 // Bad : behavior depends solely on type of the object
23 pb->f(3.14f); // Base::f(float) 3.14
24 pd->f(3.14f); // Derived::f(float) 3.14
25
26 return 0;
27 }
(1)一个函数在基类申明一个virtual,那么在所有的派生类都是是virtual的。
(2)一个函数在基类为普通函数,在派生类定义为virtual的函数称为越位,函数行为依赖于指针/引用的类型,而不是实际对象的类型。
1 #include<iostream>
2 using namespace std;
3
4 class Base
5 {
6 public:
7 void f(){ cout << "Base::f() " << endl; }
8 virtual void g(){ cout << "Base::g() " << endl; }
9 };
10
11 class Derived : public Base
12 {
13 public:
14 virtual void f(){ cout << "Derived::f() " << endl; }
15 void g(){ cout << "Derived::g() " << endl; }
16 };
17
18 class VirtualDerived : virtual public Base
19 {
20 public:
21 void f(){ cout << "VirtualDerived::f() " << endl; }
22 void g(){ cout << "VirtualDerived::g() " << endl; }
23 };
24
25 int main()
26 {
27 Base *d = new Derived;
28 Base *vd = new VirtualDerived;
29
30 d->f(); // Base::f() Bad behavior
31 d->g(); // Derived::g()
32
33 vd->f(); // Base::f() Bad behavior
34 vd->g(); // VirtualDerived::g()
35
36 delete d;
37 delete vd;
38
39 return 0;
40 }
《Effective C++》条款: 决不要重新定义继承而来的非虚函数。说明了不能重新定义继承而来的非虚函数的理论依据是什么。
以下摘自《Effective C++》:
公有继承的含义是 "是一个","在一个类中声明一个非虚函数实际上为这个类建立了一种特殊性上的不变性"。如果将这些分析套用到类B、类D和非虚成员函数B::mf,那么:
(1)适用于B对象的一切也适用于D对象,因为每个D的对象 "是一个" B的对象。
(2)B的子类必须同时继承mf的接口和实现,因为mf在B中是非虚函数。
那么,如果D重新定义了mf,设计中就会产生矛盾。如果D真的需要实现和B不同的mf,而且每个B的对象(无论怎么特殊)也真的要使用B实现的mf,那么每个D将不 "是一个" B。这种情况下,D不能从B公有继承。相反,如果D真的必须从B公有继承,而且D真的需要和B不同的mf的实现,那么,mf就没有为B反映出特殊性上的不变性。这种情况下,mf应该是虚函数。最后,如果每个D真的 "是一个" B,并且如果mf真的为B建立了特殊性上的不变性,那么,D实际上就不需要重新定义mf,也就决不能这样做。
不管采用上面的哪一种论据都可以得出这样的结论:任何条件下都要禁止重新定义继承而来的非虚函数。