• 【C++进阶学习】第二弹——继承(下)——挖掘继承深处的奥秘


    继承(上):【C++进阶学习】第一弹——继承(上)——探索代码复用的乐趣-CSDN博客

    前言:

    在前面我们已经讲了继承的基础知识,让大家了解了一下继承是什么,但那些都不是重点,今天,我们一起来挖掘一下继承底层的一些知识和一些极容易出错的点

    目录

    一、隐藏

    1.1 隐藏的概念

    1.2 隐藏的两种类型

    二、派生类的默认成员函数

    三、继承与友元

    四、继承与静态成员

    五、总结


    一、隐藏

    1.1 隐藏的概念

    在 C++ 中,继承是一种机制,使得子类可以继承父类的成员变量和成员函数。然而,当子类中出现和父类同名的成员变量或成员函数时,会发生一种特殊的现象,即隐藏

    隐藏是指:如果子类中出现了与父类同名的成员变量或成员函数,则子类中的这个成员会“隐藏”父类中的同名成员,使得父类中的同名成员在子类中不可见。

    1.2 隐藏的两种类型

    具体来说,有以下两种情况:

    成员变量隐藏:
    如果子类中出现了和父类同名的成员变量,则子类中的这个成员变量会隐藏父类中的同名成员变量。例如:

    1. class Parent {
    2. public:
    3.     int a;
    4. };
    5. class Child : public Parent {
    6. public:
    7.     int a; // 此处的 a 会隐藏 Parent 中的 a
    8. };
    9. int main() {
    10.     Child c;
    11.     c.a = 10; // 此处修改的是 Child 中的 a,而不是 Parent 中的 a
    12.     return 0;
    13. }

    成员函数隐藏:
    如果子类中出现了和父类同名的成员函数,则子类中的这个成员函数会隐藏父类中的同名成员函数。例如:

    1. class Parent {
    2. public:
    3.     void func() {
    4.         cout << "Parent::func()" << endl;
    5.     }
    6. };
    7. class Child : public Parent {
    8. public:
    9.     void func() {
    10.         cout << "Child::func()" << endl;
    11.     }
    12. };
    13. int main() {
    14.     Child c;
    15.     c.func(); // 此处调用的是 Child 中的 func(),而不是 Parent 中的 func()
    16.     return 0;
    17. }

    需要注意的是,虽然子类中的成员会隐藏父类中的同名成员,但是父类中的成员仍然存在,只是在子类中不可见。如果想在子类中访问父类中被隐藏的成员,可以使用作用域运算符(::)来显式地指明要访问的成员所在的类。例如:

    1. class Parent {
    2. public:
    3.     int a;
    4. };
    5. class Child : public Parent {
    6. public:
    7.     int a;
    8. };
    9. int main() {
    10.     Child c;
    11.     c.a = 10; // 此处修改的是 Child 中的 a
    12.     c.Parent::a = 20; // 此处修改的是 Parent 中的 a
    13.     return 0;
    14. }

    总之,在 C++ 中的继承中,隐藏是一种特殊的机制,需要注意避免误用。

    二、派生类的默认成员函数

    在 C++ 中,当我们定义一个类时,可以省略掉其中的成员函数的实现,而直接在类定义的外部提供其实现。这种情况下,如果我们不提供任何实现,那么 C++ 编译器会自动为我们提供一个默认的构造函数、析构函数和拷贝构造函数和拷贝赋值运算符。

    对于派生类来说,情况也是类似的。当我们定义一个派生类时,如果我们不提供任何构造函数,那么 C++ 编译器会自动为我们提供一个默认的构造函数,其构造函数的参数列表和父类的构造函数一致。例如:

    1. class Parent {
    2. public:
    3.     Parent(int a) : m_a(a) {}
    4.     int m_a;
    5. };
    6. class Child : public Parent {
    7. public:
    8.     Child() : Parent(0) {} // 此处的构造函数会自动调用 Parent 的构造函数,并传入 0 作为参数
    9. };

    同时,如果我们没有提供任何析构函数,那么 C++ 编译器也会自动为我们提供一个默认的析构函数,其析构函数的函数体为空。例如:

    1. class Parent {
    2. public:
    3.     ~Parent() {}
    4. };
    5. class Child : public Parent {
    6. public:
    7.     ~Child() {} // 此处的析构函数会自动调用 Parent 的析构函数
    8. };

    需要注意的是,如果我们提供了任何一个构造函数或析构函数,那么 C++ 编译器就不会再为我们提供默认的构造函数或析构函数了。这时如果我们需要使用默认的构造函数或析构函数,需要我们自己显式地提供。(析构顺序为先派生类再基类)

    另外,对于拷贝构造函数和拷贝赋值运算符来说,如果我们没有提供任何拷贝构造函数和拷贝赋值运算符,那么 C++ 编译器会自动为我们提供一个默认的拷贝构造函数和拷贝赋值运算符,其行为是浅拷贝(Shallow Copy),即直接拷贝成员变量的值。例如:

    1. class Parent {
    2. public:
    3.     Parent(int a) : m_a(a) {}
    4.     int m_a;
    5. };
    6. class Child : public Parent {
    7. public:
    8.     Child(int a, int b) : Parent(a), m_b(b) {}
    9.     int m_b;
    10. };
    11. int main() {
    12.     Child c1(1, 2);
    13.     Child c2(c1); // 此处调用的是 Child 的默认拷贝构造函数,直接拷贝 m_a 和 m_b 的值
    14.     return 0;
    15. }

    但是,如果我们的类中有指针类型的成员变量,那么默认的拷贝构造函数和拷贝赋值运算符就会出现问题,因为它们只会拷贝指针的值,而不会拷贝指针所指向的内存。这时我们需要自己提供一个拷贝构造函数和拷贝赋值运算符,实现深拷贝(Deep Copy)。例如:

    1. class Parent
    2. {
    3. public:
    4. Parent(int a)
    5. :_a(a)
    6. {}
    7. int _a;
    8. };
    9. class Child :public Parent
    10. {
    11. public:
    12. int* _b;
    13. Child(int a, int b)
    14. :Parent(a),
    15. _b(new int(b)) //深度拷贝
    16. {}
    17. ~Child()
    18. {
    19. delete _b;
    20. _b = nullptr;
    21. }
    22. Child(const Child& c)
    23. :Parent(c)
    24. {
    25. _b = new int(*c._b);
    26. }
    27. Child& operator=(const Child& c)
    28. {
    29. if (this != &c)
    30. {
    31. _a = c._a;
    32. delete _b;
    33. _b = new int(*(c._b));
    34. }
    35. return *this;
    36. }
    37. void Print()
    38. {
    39. cout << "_a:" << _a << " " << "_b:" << *_b << endl;
    40. }
    41. };
    42. int main()
    43. {
    44. Child c1(1, 2);
    45. Child c2(c1); //此处调用的是c2的深度拷贝
    46. c2.Print();
    47. Child c3 = c1;
    48. c3.Print();
    49. return 0;
    50. }

    运行结果:

    我们通过两张图来总结一下:

    三、继承与友元

    在C++中,友元关系不能被继承,因为友元关系是独立于类定义的,并不是类的成员。因此,如果在父类中声明了一个友元函数或友元类,子类无法继承这种关系。

    下面是一些相关知识点:

    1、友元函数不能是成员函数:友元函数不是类的成员函数,因此不能使用this指针,也不能直接访问类的私有成员。需要通过类的对象或引用来访问私有成员。
    2、友元关系不具有传递性:如果A类声明了B类为友元,则B类不会自动成为A类的友元。
    友元函数可以是模板函数:模板函数可以被声明为类的友元,这样模板函数可以访问类的私有成员。
    3、友元类:如果一个类声明另一个类为友元,则该友元类的所有成员函数都可以访问原类的私有成员。
    4、友元不能继承:由于友元关系不是类的成员,因此不能被继承。如果在父类中声明了一个友元函数或友元类,子类无法继承这种关系。但子类可以在自己的范围内重新声明该友元关系。
    示例:

    1. #include
    2. #include
    3. using namespace std;
    4. class Base {
    5. public:
    6. friend void friendFunction(Base&);
    7. protected:
    8. int value;
    9. };
    10. class Derived : public Base {
    11. public:
    12. // friendFunction 在 Derived 中不是友元函数,需要重新声明
    13. friend void friendFunction(Derived&);
    14. };
    15. void friendFunction(Base& base) {
    16. // 可以访问 Base 的私有成员
    17. base.value = 10;
    18. }
    19. void friendFunction(Derived& derived) {
    20. // 可以访问 Derived 的私有成员
    21. derived.value = 20;
    22. }
    23. int main() {
    24. Derived derived;
    25. friendFunction(derived);
    26. return 0;
    27. }

    在上面的示例中,由于友元关系不能被继承,因此在Derived类中需要重新声明friendFunction函数为友元函数,以便在Derived类的实例上调用该函数。

    四、继承与静态成员

    在 C++ 中,类可以包含静态成员变量和静态成员函数,其中静态成员变量属于类本身,而不是类的某个对象,因此它们可以在不创建类对象的情况下被访问。

    当一个类继承另一个类时,子类可以继承其父类的静态成员,并且可以在子类中重新定义这些静态成员。在这种情况下,子类和父类各自拥有自己的静态成员变量,它们之间没有任何关系。

    下面是一个简单的例子:

    1. #include
    2. using namespace std;
    3. class Parent {
    4. public:
    5. static int a;
    6. };
    7. int Parent::a = 10; //静态成员的定义只能在类外进行
    8. class Child : public Parent {
    9. public:
    10. static int a; //类中只能声明静态成员
    11. };
    12. int Child::a = 20; //静态成员的定义只能在类外进行
    13. int main() {
    14. cout << Parent::a << endl; //输出10
    15. cout << Child::a << endl; //输出20
    16. return 0;
    17. }

    运行结果:

    在上面的例子中,Parent 类和 Child 类都有一个静态成员变量 a,它们各自拥有自己的实现。在 main 函数中,我们可以直接通过类名来访问这些静态成员变量。

           需要注意的是,如果子类中没有重新定义父类的静态成员变量,那么子类可以直接访问父类的静态成员变量,例如 Parent::a。如果子类重新定义了父类的静态成员变量,那么子类只能访问自己的静态成员变量,例如 Child::a。

           此外,静态成员函数也可以继承,并且可以在子类中重新定义。在子类中重新定义父类的静态成员函数时,子类的静态成员函数会隐藏父类的静态成员函数,因此如果在子类中需要调用父类的静态成员函数,需要使用作用域运算符 :: 来显式地调用。

           还有一个需要注意的点就是,类中只能声明静态成员,静态成员的定义只能在类外进行。

    总之,在 C++ 中,静态成员在继承中的行为与普通成员有所不同,需要注意其使用方法。

    五、总结

    以上就是C++继承中需要额外注意的点,此外,还有一个很重要的知识点我们还没讲到——多继承、菱形继承、虚拟继承,这几个知识点是有很大关联性的,且我们在平时使用继承时也很容易出错,鉴于篇幅问题,这几个问题会在下一篇单拎出来来讲,今天的内容就到此为止。

    感谢各位大佬观看,创作不易,还请各位大佬一键三连!!!

  • 相关阅读:
    深度学习与总结JVM专辑(四):类文件结构(图文+代码)
    前端小案例 | 一个带切换的登录注册界面(静态)
    C# List集合查找删除指定数据
    领航未来,2022 世界人工智能大会「元宇宙的数字原生进化」论坛等你来!
    Error: spawn webpack-dev-server EACCES
    单表数据记录查询
    MybatisPlus多表关联分页返回结果异常
    腾讯99公益日-券券松鼠❤
    【ISO】Windows10系统ISO镜像怎么从微软官网下载?
    npm 设置下载源
  • 原文地址:https://blog.csdn.net/2301_80220607/article/details/139750818