C++继承(下)

简介: C++继承(下)

继承与友元

友元关系不能继承,也就是说基类友元不能访问子类私有和保护成员。

#include<iostream>
using namespace std;
class Student;
class Person
{
public:
  friend void Display(const Person& p, const Student& s);
protected:
  string _name; // 姓名
};
class Student : public Person
{
protected:
  int _stuNum; // 学号
};
void Display(const Person& p, const Student& s)
{
  cout << p._name << endl;
  cout << s._stuNum << endl;
}
void main()
{
  Person p;
  Student s;
  Display(p, s);
}

继承与静态成员

基类定义了static静态成员,则整个继承体系里面只有一个这样的成员。无论派生出多少个子

类,都只有一个static成员实例 。

#include<iostream>
using namespace std;
class Person
{
public:
  Person() { ++_count; }
protected:
  string _name; // 姓名
public:
  static int _count; // 统计人的个数。
};
int Person::_count = 0;
class Student : public Person
{
protected:
  int _stuNum; // 学号
};
int main()
{
  Person s1;
  Student s2;
  cout << s1._count << endl;
  cout << s2._count << endl;
  cout << &s1._count << endl;
  cout << &s2._count << endl;
  return 0;
}

这是因为静态成员存储的区域不一样,是储存在静态区的。

这里还要注意一种情况:

这样子是可以的,因为虽然是*和->但是不会去真正的在这个对象里面找,因为静态区是不在这个对象中的,就像之前的成员函数一样,定义的内容也不是在对象内部,只是声明在类中,传递过去的也是空指针,并没有真正的去访问空指针。

其实这样子就相当于告诉你去哪个类里面找。

多继承

一个类继承了多个个类,这就是多继承。

菱形继承

最麻烦的就是这种菱形继承,因为数据会冗余。

#include<iostream>
using namespace std;
class Person
{
public:
  string _name; // 姓名
};
class Student : public Person
{
protected:
  int _num; //学号
};
class Teacher : public Person
{
protected:
  int _id; // 职工编号
};
class Assistant : public Student, public Teacher
{
protected:
  string _majorCourse; // 主修课程
};
int main()
{
  Assistant s;
  //s._name;//编译器不知道我们要访问哪个name,产生了二义性
  s.Student::_name = "xxx";
  s.Teacher::_name = "yyy";
  return 0;
}

虚继承

但是指定作用域去访问并没有彻底的解决这里的问题,所以有了一个关键字,virtual。

虚继承要在菱形继承中间的位置。

#include<iostream>
using namespace std;
class Person
{
public:
  string _name; // 姓名
};
class Student : virtual public Person
{
protected:
  int _num; //学号
};
class Teacher : virtual public Person
{
protected:
  int _id; // 职工编号
};
class Assistant : public Student, public Teacher
{
protected:
  string _majorCourse; // 主修课程
};
int main()
{
  Assistant s;
  s._name = "xxx";//编译器不知道我们要访问哪个name
  //s.Student::_name = "xxx";
  //s.Teacher::_name = "yyy";
  return 0;
}

现在这三个域中的name就都是同一个name了。

虚继承的原理

先来看看普通类,因为编译器内部做了处理,这里只能用内存窗口去观察了。

#include<iostream>
using namespace std;
class A
{
public:
  int _a;
};
class B : public A
//class B : virtual public A
{
public:
  int _b;
};
class C : public A
//class C : virtual public A
{
public:
  int _c;
};
class D : public B, public C
{
public:
  int _d;
};
int main()
{
  D d;
  d.B::_a = 1;
  d.C::_a = 2;
  d._b = 3;
  d._c = 4;
  d._d = 5;
  return 0;
}

这里我锁定了d地址。

然后再来看看虚继承的:

这是成员在内存中的分布位置。

那么橙框中的内容又是什么呢?

这里面其实是地址,地址在调用两个内存窗口看一下:

那么这里橙色框里面又是什么。

我们看一下距离,第一个距离20,第二个距离12,也就是说上面橙框里面的是找到虚基类对象的偏移量!

那么为什么要储存偏移量呢?

因为子类在给父类赋值的情况下,会发生切片,虚继承的成员因为是在最下面,所以中间如果少了内容位置就会不一样。

也就是说,虚继承之后的对象就不是在类内部了,而是放在下面某一处了,但是这个位置不确定,所以用偏移量来算才是准确的。

像最后一个赋值给父类,他又指向了原来d对象的位置,进入之后也是通过地址+偏移量找到虚继承对象。

那么虚继承看起来好像更加占用空间了,那么如果虚继承对象的成员很大,不用虚继承会有多份,但是用了虚继承就只有一份。

存偏移量的地方叫做虚继表,对象中储存了虚继表的地址。

总结

继承与组合:

//继承
class A
{
  int _a;
};
class B:public A
{
  int _b;
};
//组合
class X
{
  int _x;
};
class Y
{
  X x;
  int _y;
};

这两种都是复用。

但是继承是保护能用,组合不能用,所有就有人提出了两个概念。

继承叫做白箱复用,组合叫黑箱复用。

白箱意思就是能看清楚里面的内容,根据某些底层原理去实现对应的功能,黑箱是不注重底层的内容,只要能完成任务就行。

这里还有一个区别,继承的耦合度比组合的高。

我们一般追求的都是低耦合。

继承当中改了父类的内容,子类的内容大概率也会被修改,组合的概率就会小,因为继承跟保护和公有成员有联系,组合只和公有成员有联系。

继承:例如学生和人都有很多共同特征,可以说学生是一个人,这个人是学生。

组合:例如头和眼睛,眼睛应该在头上,但是他们细节没相似之处,不能说头是眼睛。

继承和组合要根据实际情况去使用,如果都差不多就用组合。

相关文章
|
1月前
|
安全 Java 编译器
C++进阶(1)——继承
本文系统讲解C++继承机制,涵盖继承定义、访问限定符、派生类默认成员函数、菱形虚拟继承原理及组合与继承对比,深入剖析其在代码复用与面向对象设计中的应用。
|
5月前
|
存储 安全 Java
c++--继承
c++作为面向对象的语言三大特点其中之一就是继承,那么继承到底有何奥妙呢?继承(inheritance)机制是面向对象程序设计使代码可以复用的最重要的手段,它允许程序员在保持原有类特性的基础上进行扩展,增加功能,这样产生新的类,称派生类。继承呈现了面向对象程序设计的层次结构,体现了由简单到复杂的认知过程。以前我们接触的复用都是函数复用,继承是类设计层次的复用,继承就是类方法的复用。
128 0
|
8月前
|
安全 C++
【c++】继承(继承的定义格式、赋值兼容转换、多继承、派生类默认成员函数规则、继承与友元、继承与静态成员)
本文深入探讨了C++中的继承机制,作为面向对象编程(OOP)的核心特性之一。继承通过允许派生类扩展基类的属性和方法,极大促进了代码复用,增强了代码的可维护性和可扩展性。文章详细介绍了继承的基本概念、定义格式、继承方式(public、protected、private)、赋值兼容转换、作用域问题、默认成员函数规则、继承与友元、静态成员、多继承及菱形继承问题,并对比了继承与组合的优缺点。最后总结指出,虽然继承提高了代码灵活性和复用率,但也带来了耦合度高的问题,建议在“has-a”和“is-a”关系同时存在时优先使用组合。
446 6
|
10月前
|
C++ 开发者
C++学习之继承
通过继承,C++可以实现代码重用、扩展类的功能并支持多态性。理解继承的类型、重写与重载、多重继承及其相关问题,对于掌握C++面向对象编程至关重要。希望本文能为您的C++学习和开发提供实用的指导。
156 16
|
10月前
|
编译器 数据安全/隐私保护 C++
【C++面向对象——继承与派生】派生类的应用(头歌实践教学平台习题)【合集】
本实验旨在学习类的继承关系、不同继承方式下的访问控制及利用虚基类解决二义性问题。主要内容包括: 1. **类的继承关系基础概念**:介绍继承的定义及声明派生类的语法。 2. **不同继承方式下对基类成员的访问控制**:详细说明`public`、`private`和`protected`继承方式对基类成员的访问权限影响。 3. **利用虚基类解决二义性问题**:解释多继承中可能出现的二义性及其解决方案——虚基类。 实验任务要求从`people`类派生出`student`、`teacher`、`graduate`和`TA`类,添加特定属性并测试这些类的功能。最终通过创建教师和助教实例,验证代码
263 5
|
12月前
|
编译器 C++ 开发者
【C++】继承
C++中的继承是面向对象编程的核心特性之一,允许派生类继承基类的属性和方法,实现代码复用和类的层次结构。继承有三种类型:公有、私有和受保护继承,每种类型决定了派生类如何访问基类成员。此外,继承还涉及构造函数、析构函数、拷贝构造函数和赋值运算符的调用规则,以及解决多继承带来的二义性和数据冗余问题的虚拟继承。在设计类时,应谨慎选择继承和组合,以降低耦合度并提高代码的可维护性。
224 1
【C++】继承
|
安全 程序员 编译器
【C++篇】继承之韵:解构编程奥义,领略面向对象的至高法则
【C++篇】继承之韵:解构编程奥义,领略面向对象的至高法则
196 11
|
C++
C++番外篇——对于继承中子类与父类对象同时定义其析构顺序的探究
C++番外篇——对于继承中子类与父类对象同时定义其析构顺序的探究
152 1
|
C++
C++番外篇——虚拟继承解决数据冗余和二义性的原理
C++番外篇——虚拟继承解决数据冗余和二义性的原理
146 1
C++(二十)继承
本文介绍了C++中的继承特性,包括公有、保护和私有继承,并解释了虚继承的作用。通过示例展示了派生类如何从基类继承属性和方法,并保持自身的独特性。此外,还详细说明了派生类构造函数的语法格式及构造顺序,提供了具体的代码示例帮助理解。