隐藏/重定义:子类和父类有同名的成员,子类成员隐藏了父类的成员。
重载:同一个作用域,重载了参数。
(在实际中最好不要定义同名函数)
子类对象不能初始化父类对象,用父类成员初始化子类成员。
析构函数会自动调用。
显示调用父类析构,无法保证先子后父,所以子类析构函数完成舅自定义调用父类析构,这样就保证了先子后父。
友元关系不能继承。
子类继承父类的静态成员,继承的是使用权(地址一样)
面经:
1.
什么是菱形继承?菱形继承的问题是什么?
答:菱形继承(Diamond Inheritance)是指在多继承的编程语言中出现的一种特定的继承模式。这种模式涉及到四个类:一个基类(A),两个派生类(B 和 C),以及一个从这两个派生类继承的顶层类(D)。
问题:数据冗余、二义性。
二义性:当两个派生类 B 和 C 都覆盖或使用了来自基类 A 的某个成员时,顶层类 D 将不知道应该使用哪一个版本,因为每个派生类都有自己的 A 类的副本。例如,如果 A 有一个方法
func()
, B 和 C 都继承并可能修改了这个方法,那么 D 中调用
func()
时就会产生疑问:它应该调用哪个版本?
二义性解决办法:
C++:使用虚继承(virtual inheritance)。虚继承确保了即使有多个派生路径,也只有一个基类实例存在。这样可以避免二义性问题。
2.
什么是菱形虚拟继承?如何解决数据冗余和二义性的
答:菱形虚拟继承是C++中解决菱形继承问题的一种机制。菱形继承是指一个类同时从两个类继承,而这两个类又共同从另一个类继承的情况。这种继承模式可能会导致数据冗余和二义性问题。
如何实现菱形虚拟继承
为了使用菱形虚拟继承,你需要在中间层类(即图中的B和C)中将对基类A的继承声明为虚拟继承。下面是一个例子:
class A {
public:
int data;
};
class B : virtual public A { /* ... */ };
class C : virtual public A { /* ... */ };
class D : public B, public C {
public:
void setData(int value) { data = value; } // 此处的 data 不会产生二义性
};
在这个例子中,B
和 C
类都使用 virtual public A
来声明对 A
类的继承。这确保了尽管 D
类从 B
和 C
继承,但是 A
类的数据成员 data
只会被继承一次。
- 使用
virtual
关键字:在中间层类(如B和C)中使用virtual
关键字修饰继承。 - 仅在必要时使用:虚拟继承会增加类层次结构的复杂度,并且可能会影响性能(因为它需要额外的间接性来支持虚拟基类的唯一实例),因此只在需要时使用。
- 虚拟继承的影响:虚拟继承不仅解决了数据冗余和二义性问题,还会影响构造函数和析构函数的调用顺序,因此需要特别注意这一点。
3.
继承和组合的区别?什么时候用继承?什么时候用组合?
答:
很多人说
C++
语法复杂,其实多继承就是一个体现。有了多继承
,就存在菱形继承,有了菱形继承就有菱形虚拟继承,底层实现就很复杂。所以一般不建议设计出多继承,一定不要设计出菱形继承。否则在复杂度及性能上都有问题。
多继承可以认为是
C++
的缺陷之一,很多后来的
OO
语言都没有多继承,如
Java
。
继承和组合
public
继承是一种
is-a
的关系。也就是说每个派生类对象都是一个基类对象。
组合是一种
has-a
的关系。假设
B
组合了
A
,每个
B
对象中都有一个
A
对象。
优先使用对象组合,而不是类继承
。
继承允许你根据基类的实现来定义派生类的实现。这种通过生成派生类的复用通常被称为白箱复用(white-box reuse)
。术语
“
白箱
”
是相对可视性而言:在继承方式中,基类的内部细节对子类可见 。继承一定程度破坏了基类的封装,基类的改变,对派生类有很大的影响。派生类和基类间的依赖关系很强,耦合度高。
对象组合是类继承之外的另一种复用选择。新的更复杂的功能可以通过组装或组合对象来获得。对象组合要求被组合的对象具有良好定义的接口。这种复用风格被称为黑箱复用(black-box reuse)
,因为对象的内部细节是不可见的。对象只以
“
黑箱
”
的形式出现。
组合类之间没有很强的依赖关系,耦合度低。优先使用对象组合有助于你保持每个类被封装。
实际尽量多去用组合。组合的耦合度低,代码维护性好。不过继承也有用武之地的,有些关系就适合继承那就用继承,另外要实现多态,也必须要继承。类之间的关系可以用继承,可以用组合,就用组合。