类成员函数的重载、覆盖和隐藏区别 (C++)
这是本人第一次写博客,主要是想记录自己的学习过程、心得体会,一是可以方便以后回顾相关知识,二是可以与大家相互学习交流。
关于C++中类成员函数的重载、覆盖和隐藏区别,第一次看到这个问题是在准备找工作的时候在牛客网中(http://www.nowcoder.com/questionTerminal/266d3a6d4f1b436aabf1eff3156fed95)看到的,后来也有遇到一些题在考察这个知识点,只要理解了他们的区别,还是可以正确的解答有关这个知识点的题。看到牛客网中有大牛的解释很好,故摘录保存。
1.成员函数被重载的特征:overload
(1)相同的范围(在同一个类中);
(2)函数名字相同;
(3)参数不同;
(4)virtual 关键字可有可无。
2.覆盖是指派生类函数覆盖基类函数,特征是:override/overwrite
(1)不同的范围(分别位于派生类与基类);
(2)函数名字相同;
(3)参数相同;
(4)基类函数必须有virtual 关键字。
3.“隐藏”是指派生类的函数屏蔽了与其同名的基类函数,规则如下:hide
(1)如果派生类的函数与基类的函数同名,但是参数不同。此时,不论有无virtual关键字,基类的函数将被隐藏(注意别与重载混淆:作用域不同)。
(2)如果派生类的函数与基类的函数同名,并且参数也相同,但是基类函数没有virtual 关键字。此时,基类的函数被隐藏(注意别与覆盖混淆:有没有)。
有关重载,应该是很好理解的(在同一个类中,函数名相同,函数的参数类型和个数不同),此处不予更多的说明,对于覆盖和隐藏(主要在基类和派生类中体现),借助下面这个例子来直观的说明:
1 #include<iostream> 2 using namespace std; 3 4 class People { 5 public: 6 // 隐藏:是指派生类的函数屏蔽基类函数 7 // 隐藏规则1: 8 // 1) 函数名相同 && 参数不同 9 // 2) virtual不影响 10 void getId_different_params() {cout << "People::getId_different_params" << endl;} 11 virtual void getName_different_params() {cout << "People::getName_different_params" << endl;} 12 13 // 隐藏规则2: 14 // 1) 函数名相同 && 参数相同 15 // 2) 无virtual 16 void getPhone_same_params() {cout << "People::getPhone_same_params" << endl;} 17 18 // 覆盖规则: 19 // 1) 函数名相同 && 参数相同 20 // 2) 有virtual 21 virtual void getAddress_same_params() {cout << "People::getAddress_same_params" << endl;} 22 }; 23 24 class Children : public People { 25 public: 26 // 隐藏:是指派生类的函数屏蔽基类函数 27 // 隐藏规则1: 28 // 1) 函数名相同 && 参数不同 29 // 2) virtual不影响 30 void getId_different_params(int) {cout << "Children::getId_different_params(int)" << endl;} 31 virtual void getName_different_params(int) {cout << "Children::getName_different_params(int)" << endl;} 32 33 // 隐藏规则2: 34 // 1) 函数名相同 && 参数相同 35 // 2) 无virtual 36 void getPhone_same_params() {cout << "Children::getPhone_same_params" << endl;} 37 38 // 覆盖规则: 39 // 1) 函数名相同 && 参数相同 40 // 2) 有virtual 41 virtual void getAddress_same_params() {cout << "Children::getAddress_same_params" << endl;} 42 }; 43 44 int main() 45 { 46 Children *c = new Children(); 47 c->getId_different_params(1); 48 /*由于被隐藏,所以编译时找不到对应函数 49 c->getName_different_params(); 50 c->getId_different_params(); 51 */ 52 c->getName_different_params(1); 53 c->getId_different_params(); 54 c->getName_different_params(); 55 c->getPhone_same_params(); 56 c->getAddress_same_params(); 57 /*输出为: 58 Children::getId_different_params(int) 59 Children::getName_different_params(int) 60 Children::getPhone_same_params 61 Children::getAddress_same_params 62 */ 63 64 People *p = new Children(); 65 p->getId_different_params(); 66 p->getName_different_params(); 67 p->getPhone_same_params(); 68 p->getAddress_same_params(); 69 70 /*输出为: 71 People::getId_different_params //由于子类中的函数只是隐藏了基类中的函数而没有覆盖,因为指针为基类指针,故调用基类函数 72 People::getName_different_params //由于子类中的函数只是隐藏了基类中的函数而没有覆盖,因为指针为基类指针,故调用基类函数 73 People::getPhone_same_params //由于子类中的函数只是隐藏了基类中的函数而没有覆盖,因为指针为基类指针,故调用基类函数 74 Children::getAddress_same_params // 由于子类中的函数覆盖了基类中的函数,虽然指针为基类指针,但是会调用子类函数 75 */ 76 77 return 0; 78 }