黑马程序员C++类和对象【6】—— 继承(一文万字带你搞懂C++继承 —— 你还不知道怎么学C++继承吗?不知道虚继承底层原理吗?)(2)

简介: 黑马程序员C++类和对象【6】—— 继承(一文万字带你搞懂C++继承 —— 你还不知道怎么学C++继承吗?不知道虚继承底层原理吗?)(2)

继承同名成员处理方式

问题:

当子类与父类出现同名的成员,如何通过子类对象,访问到子类或父类中同名的数据呢?

  • 访问子类同名成员 直接访问即可(就近原则,正常访问的是子类的)
  • 访问父类同名成员 需要加作用域
class Base {
public:
  Base()
  {
    m_A = 100;
  }
  void func()
  {
    cout << "Base - func()调用" << endl;
  }
  void func(int a)
  {
    cout << "Base - func(int a)调用" << endl;
  }
public:
  int m_A;
};
class Son : public Base {
public:
  Son()
  {
    m_A = 200;
  }
  //如果想访问父类中被隐藏的同名成员函数,需要加父类的作用域
  void func()
  {
    cout << "Son - func()调用" << endl;
  }
public:
  int m_A;
};
void test01()
{
  Son s;
  cout << "Son下的m_A = " << s.m_A << endl;
  cout << "Base下的m_A = " << s.Base::m_A << endl;
  s.func();
  s.Base::func();
  //当子类与父类拥有同名的成员函数,子类会隐藏父类中所有版本的同名成员函数(只要是func的全部隐藏,不能直接访问),想要访问只能加作用域。
    s.func(100);    //报错
  s.Base::func(10);
}
int main() {
  test01();
  system("pause");
  return EXIT_SUCCESS;
}

总结:

  1. 子类对象可以直接访问到子类中同名成员
  2. 子类对象加作用域可以访问到父类同名成员
  3. 当子类与父类拥有同名的成员函数,子类会隐藏父类中同名成员函数,加作用域可以访问到父类中同名函数

继承同名静态成员处理方式

问题:

       继承中同名的静态成员在子类对象上如何进行访问?

静态成员和非静态成员出现同名,处理方式一致

  • 访问子类同名成员 直接访问即可
  • 访问父类同名成员 需要加作用域

示例:

class Base {
public:
  static void func()
  {
    cout << "Base - static void func()" << endl;
  }
  static void func(int a)
  {
    cout << "Base - static void func(int a)" << endl;
  }
  //静态成员属性(编译阶段分配到内存,所有对象共享同一份数据)
  static int m_A;
};
//类内声明,类外初始化
int Base::m_A = 100;
class Son : public Base {
public:
  static void func()
  {
    cout << "Son - static void func()" << endl;
  }
  static int m_A;
};
int Son::m_A = 200;
//同名成员属性
void test01()
{
  //通过对象访问
  cout << "通过对象访问: " << endl;
  Son s;
  cout << "Son  下 m_A = " << s.m_A << endl;
  cout << "Base 下 m_A = " << s.Base::m_A << endl;
  //通过类名访问(静态成员变量专属福利)
  cout << "通过类名访问: " << endl;
  cout << "Son  下 m_A = " << Son::m_A << endl;
    //第一个::代表通过类目方式访问,第二个::代表访问父类作用域下
  cout << "Base 下 m_A = " << Son::Base::m_A << endl;
}
//同名成员函数
void test02()
{
  //通过对象访问
  cout << "通过对象访问: " << endl;
  Son s;
  s.func();
  s.Base::func();
  cout << "通过类名访问: " << endl;
  Son::func();
  Son::Base::func();
  //出现同名,子类会隐藏掉父类中所有同名成员函数,需要加作作用域访问
  Son::Base::func(100);
}
int main() {
  //test01();
  test02();
  system("pause");
  return 0;
}

总结:

同名静态成员处理方式和非静态处理方式一样,只不过有两种访问的方式(通过对象 和 通过类名)

多继承语法

C++允许一个类继承多个类

语法:

   class 子类 :继承方式 父类1 , 继承方式 父类2...

多继承可能会引发父类中有同名成员出现,需要加作用域区分

       C++实际开发中不建议用多继承

示例:

class Base1 {
public:
  Base1()
  {
    m_A = 100;
  }
public:
  int m_A;
};
class Base2 {
public:
  Base2()
  {
    m_A = 200;  //开始是m_B 不会出问题,但是改为mA就会出现不明确
  }
public:
  int m_A;
};
//子类需要继承Base1和Base2
//语法:class 子类:继承方式 父类1 ,继承方式 父类2 …
class Son : public Base2, public Base1 
{
public:
  Son()
  {
    m_C = 300;
    m_D = 400;
  }
public:
  int m_C;
  int m_D;
};
//多继承容易产生成员同名的情况
//通过使用类名作用域可以区分调用哪一个基类的成员
void test01()
{
  Son s;
    //16    也可以使用开发者工具验证(前面已经说了)
  cout << "sizeof Son = " << sizeof(s) << endl;
    //当父类中出现同名成员,需要加作用域区分
  cout << s.Base1::m_A << endl;
  cout << s.Base2::m_A << endl;
}
int main() {
  test01();
  system("pause");
  return 0;
}

sizeof的结果:

总结:

       多继承中如果父类中出现了同名情况,子类使用时候要加作用域

菱形继承(C++特色)

菱形继承概念:

       两个派生类继承同一个基类

       又有某个类同时继承者两个派生类

       这种继承被称为菱形继承,或者钻石继承

菱形继承问题:

  1. 羊继承了动物的数据,驼同样继承了动物的数据,当草泥马使用数据时,就会产生二义性。
  2. 草泥马继承自动物的数据继承了两份,其实我们应该清楚,这份数据我们只需要一份就可以。
//动物类
class Animal
{
public:
  int m_Age;
};
//利用虚继承,解决菱形继承问题。
//继承前加virtual关键字后,变为虚继承
//此时公共的父类Animal称为虚基类
class Sheep : virtual public Animal {};        //羊类
class Tuo   : virtual public Animal {};        //驼类
class SheepTuo : public Sheep, public Tuo {};  //羊驼类
void test01()
{
  SheepTuo st;
  st.Sheep::m_Age = 10;
  st.Tuo::m_Age = 20;
    //当出现菱形继承的时候,两个父类拥有系统数据,需要加以作用域区分
  cout << "st.Sheep::m_Age = " << st.Sheep::m_Age << endl;
  cout << "st.Tuo::m_Age = " <<  st.Tuo::m_Age << endl;
    //这份数据我们知道,只要有一份就行了,菱形继承导致数据有两份,资源浪费。
  cout << "st.m_Age = " << st.m_Age << endl;
}
int main() {
  test01();
  system("pause");
  return 0;
}

菱形继承的问题二:

加virtual虚继承之前:

利用虚继承,解决菱形继承的问题。

总结:

  • 菱形继承带来的主要问题是子类继承两份相同的数据,导致资源浪费以及毫无意义
  • 利用虚继承可以解决菱形继承问题

虚继承解决菱形继承问题的底层原理图解:

虚继承解决菱形继承问题的底层剖析                      —— By CSDN码雨

结语:

       博客制作不易,画图有点费眼,要是觉得博客对您有帮助,请一键三连嗷!

                                                                                                     —— By CSDN码雨


相关文章
|
8月前
|
缓存 算法 程序员
C++STL底层原理:探秘标准模板库的内部机制
🌟蒋星熠Jaxonic带你深入STL底层:从容器内存管理到红黑树、哈希表,剖析迭代器、算法与分配器核心机制,揭秘C++标准库的高效设计哲学与性能优化实践。
C++STL底层原理:探秘标准模板库的内部机制
|
8月前
|
安全 Java 编译器
C++进阶(1)——继承
本文系统讲解C++继承机制,涵盖继承定义、访问限定符、派生类默认成员函数、菱形虚拟继承原理及组合与继承对比,深入剖析其在代码复用与面向对象设计中的应用。
|
存储 安全 Java
c++--继承
c++作为面向对象的语言三大特点其中之一就是继承,那么继承到底有何奥妙呢?继承(inheritance)机制是面向对象程序设计使代码可以复用的最重要的手段,它允许程序员在保持原有类特性的基础上进行扩展,增加功能,这样产生新的类,称派生类。继承呈现了面向对象程序设计的层次结构,体现了由简单到复杂的认知过程。以前我们接触的复用都是函数复用,继承是类设计层次的复用,继承就是类方法的复用。
249 0
|
编译器 C++
类和对象(中 )C++
本文详细讲解了C++中的默认成员函数,包括构造函数、析构函数、拷贝构造函数、赋值运算符重载和取地址运算符重载等内容。重点分析了各函数的特点、使用场景及相互关系,如构造函数的主要任务是初始化对象,而非创建空间;析构函数用于清理资源;拷贝构造与赋值运算符的区别在于前者用于创建新对象,后者用于已存在的对象赋值。同时,文章还探讨了运算符重载的规则及其应用场景,并通过实例加深理解。最后强调,若类中存在资源管理,需显式定义拷贝构造和赋值运算符以避免浅拷贝问题。
|
存储 编译器 C++
类和对象(上)(C++)
本篇内容主要讲解了C++中类的相关知识,包括类的定义、实例化及this指针的作用。详细说明了类的定义格式、成员函数默认为inline、访问限定符(public、protected、private)的使用规则,以及class与struct的区别。同时分析了类实例化的概念,对象大小的计算规则和内存对齐原则。最后介绍了this指针的工作机制,解释了成员函数如何通过隐含的this指针区分不同对象的数据。这些知识点帮助我们更好地理解C++中类的封装性和对象的实现原理。
|
安全 C++
【c++】继承(继承的定义格式、赋值兼容转换、多继承、派生类默认成员函数规则、继承与友元、继承与静态成员)
本文深入探讨了C++中的继承机制,作为面向对象编程(OOP)的核心特性之一。继承通过允许派生类扩展基类的属性和方法,极大促进了代码复用,增强了代码的可维护性和可扩展性。文章详细介绍了继承的基本概念、定义格式、继承方式(public、protected、private)、赋值兼容转换、作用域问题、默认成员函数规则、继承与友元、静态成员、多继承及菱形继承问题,并对比了继承与组合的优缺点。最后总结指出,虽然继承提高了代码灵活性和复用率,但也带来了耦合度高的问题,建议在“has-a”和“is-a”关系同时存在时优先使用组合。
938 6
|
编译器 C++
类和对象(下)C++
本内容主要讲解C++中的初始化列表、类型转换、静态成员、友元、内部类、匿名对象及对象拷贝时的编译器优化。初始化列表用于成员变量定义初始化,尤其对引用、const及无默认构造函数的类类型变量至关重要。类型转换中,`explicit`可禁用隐式转换。静态成员属类而非对象,受访问限定符约束。内部类是独立类,可增强封装性。匿名对象生命周期短,常用于临时场景。编译器会优化对象拷贝以提高效率。最后,鼓励大家通过重复练习提升技能!
|
安全 C语言 C++
彻底摘明白 C++ 的动态内存分配原理
大家好,我是V哥。C++的动态内存分配允许程序在运行时请求和释放内存,主要通过`new`/`delete`(用于对象)及`malloc`/`calloc`/`realloc`/`free`(继承自C语言)实现。`new`分配并初始化对象内存,`delete`释放并调用析构函数;而`malloc`等函数仅处理裸内存,不涉及构造与析构。掌握这些可有效管理内存,避免泄漏和悬空指针问题。智能指针如`std::unique_ptr`和`std::shared_ptr`能自动管理内存,确保异常安全。关注威哥爱编程,了解更多全栈开发技巧。 先赞再看后评论,腰缠万贯财进门。
649 0
|
编译器 C++ 开发者
【C++篇】深度解析类与对象(下)
在上一篇博客中,我们学习了C++的基础类与对象概念,包括类的定义、对象的使用和构造函数的作用。在这一篇,我们将深入探讨C++类的一些重要特性,如构造函数的高级用法、类型转换、static成员、友元、内部类、匿名对象,以及对象拷贝优化等。这些内容可以帮助你更好地理解和应用面向对象编程的核心理念,提升代码的健壮性、灵活性和可维护性。
|
编译器 C++ 容器
【c++11】c++11新特性(上)(列表初始化、右值引用和移动语义、类的新默认成员函数、lambda表达式)
C++11为C++带来了革命性变化,引入了列表初始化、右值引用、移动语义、类的新默认成员函数和lambda表达式等特性。列表初始化统一了对象初始化方式,initializer_list简化了容器多元素初始化;右值引用和移动语义优化了资源管理,减少拷贝开销;类新增移动构造和移动赋值函数提升性能;lambda表达式提供匿名函数对象,增强代码简洁性和灵活性。这些特性共同推动了现代C++编程的发展,提升了开发效率与程序性能。
532 12