C++重载、重写、重定义区别
来源:互联网 发布:程序员漫画 编辑:程序博客网 时间:2024/04/25 04:12
一 重载(overload)
概念:
函数有同样的名称,但是参数列表不相同的情形,这样的同名不同参数的函数之间,互相称之为重载函数。
基本条件:
- 函数名必须相同;
- 函数参数必须不相同,可以是参数类型或者参数个数不同;
- 函数返回值可以相同,也可以不相同;
注意:
- 只能通过不同的参数样式进行重载,例如:不同的参数类型,不同的参数个数,不同的参数顺序;
- 不能通过访问权限、返回类型、抛出的异常进行重载;
- 重载的函数应该在相同的作用域下;
验证程序:
12345678910111213141516171819202122232425262728293031323334353637
class A { public: void Func1(int arg1) { std::cout << "func 1" << std::endl; } // OK: 通过参数类型不同重载 Func1() void Func1(double arg1) { std::cout << "func 2" << std::endl; } // OK: 通过参数个数不同重载 Func1() void Func1(int arg1, int arg2) { std::cout << "func 3" << std::endl; } // OK: 重载函数返回值可以不同 bool Func1(int arg1, double arg2) { std::cout << "func 4" << std::endl; return true; } // ERROR: 不能只通过返回值来进行重载 /*bool Func1(int arg1) { return true; } */};int _tmain(int argc, _TCHAR* argv[]){ A a; a.Func1(1); a.Func1(1.0); a.Func1(1, 2); a.Func1(1, 2.0); return 0;}
输出:
1234
func 1func 2func 3func 4
二 重写(override)
概念:
也称为覆盖,子类重新定义父类中有相同名称和参数的虚函数,主要在继承关系中出现。
基本条件:
- 重写的函数和被重写的函数必须为virtual函数,分别位于基类和派生类中;
- 重写的函数和被重写的函数函数名和函数参数必须一致;
- 重写的函数和被重写的函数返回值相同,或者都返回指针或引用,并且派生类虚函数所返回的指针或引用的类型是基类中被替换的虚函数所返回的
- 指针或引用的类型的子类型。
注意:
- 重写的函数所抛出的异常必须和被重写的函数所抛出的异常一致,或者是其子类;
- 重写的函数的访问修饰符可以不同于被重写的函数,如基类的virtual函数的修饰符为private,派生类改为public或protected也是可以的。
- 静态方法不能被重写,也就是static和virtual不能同时使用。
- 重写的函数可以带virtual关键字,也可以不带。
验证程序:
12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273
class A {};class B : public A {};class C { public: virtual void Func1() { std::cout << "class C: func 1" << std::endl; } virtual A* Func2() { std::cout << "class C: func 2" << std::endl; return new A; } // ERROR:静态函数不能被声明为virtual,也就没办法被重写。 // static virtual void FuncStatic() {} //由于Func3被声明为private,所以需要通过public函数来调用 void ShowFunc3() { Func3(); } virtual void Func4() { std::cout << "class C: func 4" << std::endl; } private: virtual void Func3() { std::cout << "class C: func 3" << std::endl; }};class D : public C { public: // OK: 重写C类Func1,可以不带virtual关键字 void Func1() { std::cout << "class D: func 1" << std::endl; } // OK: 当返回值为指针或者引用时,返回值可以是父类返回值类型的子类 virtual B* Func2() { std::cout << "class D: func 2" << std::endl; return new B; } // ERROR: 除上面的情况,返回值类型要和父类一直 /*virtual bool Func2() { }*/ // OK: 重写的函数的访问修饰符可以不同于被重写的函数 virtual void Func3() { std::cout << "class D: func 3" << std::endl; } private: // OK: 重写的函数的访问修饰符可以不同于被重写的函数 virtual void Func4() { std::cout << "class D: func 4" << std::endl; }};int _tmain(int argc, _TCHAR* argv[]){ C* c = new D; c->Func1(); c->Func2(); c->ShowFunc3(); c->Func4(); return 0;}
输出:
1234
class D: func 1class D: func 2class D: func 3class D: func 4
三 重定义(redefining)
概念:
也叫隐藏,子类重新定义父类中的非虚函数,屏蔽了父类的同名函数
基本条件:
注意:
- 子类和父类的函数名称相同,但参数不同,此时不管父类函数是不是virtual函数,都将被隐藏。
- 子类和父类的函数名称相同,参数也相同,但是父类函数不是virtual函数,父类的函数将被隐藏。
验证程序:
12345678910111213141516171819202122232425262728293031323334353637
class A {public: void Func1(int a) { std::cout << "class A: Func1" << std::endl; } virtual void Func2(int a) { std::cout << "class A: Func2" << std::endl; }};class B : public A {public: // 将会重定义父类的方法 void Func1(int a) { std::cout << "class B: Func1" << std::endl; } // 将会重写(override)父类方法 void Func2(int a) { std::cout << "class B: Func2-1" << std::endl; } // 将会重定义父类的方法 void Func2(int a, int b) { std::cout << "class B: Func2-2" << std::endl; }};int _tmain(int argc, _TCHAR* argv[]){ B* b = new B; b->Func1(1); b->Func2(1); b->Func2(1, 1); return 0;}
输出:
123
class B: Func1class B: Func2-1class B: Func2-2
四 总结
重载、重写、重定义书面上的区别,以及各自的规则没有太大意义,而且这些名词本身都是翻译过来的,不同的地方翻译也不尽相同,如果初学C++,弄清每个概念的实际意义,以及为什么这么设计才是最重要的。