c++派生类必须显式指定public/protected/private继承方式,否则默认private导致基类成员不可访问;构造函数不继承,须在初始化列表调用基类构造;重载、隐藏、覆盖语义不同;多重继承需用virtual解决二义性与菱形继承。

派生类声明时为什么必须显式指定继承方式
不写 public、private 或 protected,C++ 默认按 private 继承——这意味着基类所有成员(哪怕原本是 public)在派生类中都变成私有,外部无法访问,连调用都报错:error: 'Base::func' is inaccessible。
常见误操作:照着 Java/C# 习惯只写 class Derived : Base,结果编译失败却找不到原因。
- 绝大多数场景用
public继承,保持接口可见性一致 -
protected继承仅当希望子类能用、但外界不能用基类接口时才考虑 -
private继承本质是“组合 + 实现复用”,不是真正意义上的“is-a”,慎用
构造函数不能被继承,但必须手动调用基类构造函数
派生类对象创建时,基类部分必须初始化;C++ 不自动调用基类构造函数,也不允许你在派生类构造函数体里用 Base(...) 调用——那会被当成普通函数调用,编译报错:error: expected primary-expression before '(' token。
正确做法只能在初始化列表里显式调用:
立即学习“C++免费学习笔记(深入)”;
class Derived : public Base {
public:
Derived(int x, int y) : Base(x), m_y(y) {} // ✅ 必须放这里
private:
int m_y;
};- 如果基类没有默认构造函数,而你又没在初始化列表中调用它,编译直接失败
- 初始化顺序固定:先基类,再成员变量(按声明顺序),和初始化列表书写顺序无关
- 虚基类构造函数由最派生类直接调用,中间层即使写了也不会执行
重载、隐藏与覆盖的区别常被混淆
这三个词对应完全不同的行为,错误理解会导致调用不到预期函数:
-
void func()在派生类里重新定义同名函数 → 隐藏(Base::func在派生类作用域不可见,除非用Base::func()显式调用) -
virtual void func()在基类声明,派生类写void func() override→ 覆盖(多态生效,通过基类指针调用会走派生类版本) - 参数列表不同(比如
func(int)vsfunc(double))→ 重载(只在同一作用域内有效;跨类不构成重载,而是隐藏)
典型坑:忘记加 virtual,或漏写 override,结果看似写了同名函数,运行时却没走新逻辑。
多重继承下二义性和菱形继承怎么处理
两个基类都有同名成员(比如都定义了 print()),派生类直接调用会报错:error: request for member 'print' is ambiguous;更麻烦的是菱形继承(Base → D1, D2 → Final),若不加 virtual,Final 会含两份 Base 子对象。
- 二义性必须用作用域解析明确指定:
d1.Base1::print()或d1.Base2::print() - 菱形继承必须让中间层用
virtual public Base,否则sizeof(Final)会翻倍,且两个Base子对象状态独立 - 虚继承带来额外开销:每个虚基类子对象多一个指针,且构造顺序更复杂(虚基类由最派生类最先构造)
虚继承不是银弹,只在真需要共享单一基类实例时才用;日常开发中,优先考虑组合或接口抽象(纯虚类)替代多重继承。
继承关系一旦变深或交叉,sizeof、构造顺序、析构顺序、指针转换(static_cast vs dynamic_cast)全都会变得敏感——别等到 core dump 才回头查内存布局。











