final是C++11起阻止类被继承的唯一标准方式,需置于类名后、左大括号前,如class Base final { }; 错误位置或用于非类类型均非法,且不具传递性。

在 C++11 及之后的标准中,final 是唯一标准、简洁且无副作用的方式,直接阻止类被继承。
如何用 final 修饰类
在类定义的末尾(即 { 之前)加上 final 关键字即可。编译器会在子类尝试继承时立即报错,错误信息通常包含 cannot derive from 或类似提示。final class
class Base final {
public:
void foo() {}
};
class Derived : public Base { // 编译错误:Base is marked final
};
-
final必须紧挨着类名后、左大括号前,位置不能错;写成class final Base { ... }或class Base { ... } final;都是非法的 - 它不改变类的访问控制、内存布局或 ABI,只是语义约束
- 模板类也可以加
final,如templateclass Container final { ... };
final 和私有析构函数的区别
过去有人用私有析构函数 + 友元模拟“不可继承”,但那是 hack,且无法真正阻止继承(子类仍可定义、实例化,只是无法安全销毁)。而 final 是编译期硬性禁止,更干净、意图更明确。
- 私有析构函数:子类能编译通过,但
delete指针或栈对象析构会失败 -
final:子类定义直接失败,连符号都生成不了 - 私有析构还会干扰
std::unique_ptr等智能指针的正常使用;final完全无此问题
注意 final 的作用域和常见误用
final 只对当前类生效,不影响其成员函数是否可被重写;若你还想禁止虚函数被重写,需单独在函数声明后加 final(如 virtual void f() final;)。
立即学习“C++免费学习笔记(深入)”;
- 不能对非类类型(如
enum class、别名、变量)使用final - 不能在类定义外部添加
final,比如在头文件声明类、源文件再补final是无效的 - 如果基类已用
final,派生类即使没写final,也不能再被继承——但这是继承链自然结果,不是final的“传递性”
真正容易被忽略的是:final 不提供运行时保护,也不影响 dynamic_cast 或 typeid 行为;它纯粹是编译器的一道语法检查闸门——只要没过这关,就根本不会有可执行的继承代码产生。










