C++适用SOLID原则但需适配其特性:SRP需结合RAII封装资源管理,避免业务类混杂生命周期逻辑;应利用值语义、移动语义和模板而非盲目套用Java式继承与shared_ptr。

C++ 里没有专属于它的 SOLID 设计原则 —— SOLID 是面向对象设计的通用原则,语言无关,C++ 完全适用,但直接套用时容易水土不服,因为 C++ 的资源管理、值语义、RAII、模板和多重继承等特性,会让某些原则的实现方式和 Java/C# 截然不同。
为什么 C++ 中 SRP(单一职责原则)要配合 RAII 理解
在 C++ 中,一个类的“职责”往往隐含资源生命周期管理。如果只按业务逻辑切分职责,却把 std::fstream 或 std::thread 的打开/关闭、启动/分离混在业务逻辑里,就实际违反了 SRP。
- 正确做法:用 RAII 封装资源(如自定义
FileGuard、LockGuard),让资源管理成为独立、可复用的类,业务类只专注领域逻辑 - 反例:
class DataProcessor { std::ofstream log_file; void process() { log_file —— 这个类同时承担数据处理 + 日志文件生命周期管理 - C++ 特有风险:析构函数抛异常会终止程序(
std::terminate),所以 RAII 类的析构必须noexcept,否则 SRP 表面成立,实则埋雷
OCP(开闭原则)在 C++ 中靠什么落地
Java/C# 依赖抽象基类 + 运行时多态,而 C++ 更倾向编译期扩展:模板 + 概念(C++20)+ CRTP,而非强制继承体系。
- 优先用策略模式的模板版本:
template,新增日志行为无需改class Processor { Logger logger; }; Processor,符合 OCP - 虚函数不是唯一路径;过度依赖
virtual会阻碍内联、增加 vtable 开销,还可能因对象切片破坏多态 - 接口类(纯虚类)在 C++ 中应尽量小而稳定,避免添加新虚函数——那会破坏所有派生类的二进制兼容性(尤其动态库场景)
LSP(里氏替换原则)在 C++ 中最常被违反的点
不是“能编译通过就能替换”,而是“行为契约不被破坏”。C++ 的值语义和 const 正确性让 LSP 更微妙。
iHuzuCMS狐族内容管理系统,是国内CMS市场的新秀、也是国内少有的采用微软的ASP.NET 2.0 + SQL2000/2005 技术框架开发的CMS,充分利用ASP.NET架构的优势,突破传统ASP类CMS的局限性,采用更稳定执行速度更高效的面向对象语言C#设计,全新的模板引擎机制, 全新的静态生成方案,这些功能和技术上的革新塑造了一个基础结构稳定功能创新和执行高效的CMS。iHuzu E
立即学习“C++免费学习笔记(深入)”;
- 子类重写虚函数时修改了前置/后置条件(比如父类允许
void set_value(int x)接受负数,子类抛异常)→ 违反 LSP - 返回类型协变仅限指针/引用,且必须保持 const 正确性:
Base* get() override可返回Derived*,但不能返回Derived&若父类返回const Base& - 更隐蔽的坑:子类重载了父类未声明为
virtual的函数,调用发生在静态绑定上下文(如模板函数内),看似替换成功,实则没走多态 → 行为不一致
依赖倒置(DIP)在 C++ 中别只盯着“抽象基类”
DIP 的本质是“依赖于稳定接口,而非易变实现”,但在 C++ 中,“稳定接口”可以是:
- 概念(
concept):约束模板参数行为,比虚函数更轻量、零成本,且编译期检查 - 非成员非友元函数:如
std::swap的 ADL 查找,解耦容器与算法 - 策略对象(传值或完美转发):
std::sort(first, last, Compare{}),依赖的是可调用对象的签名契约,不是某个基类 - 误用警告:
class Service { std::unique_ptr引入不必要的堆分配和虚调用,若db_; IDatabase仅含一两个函数,用std::function或模板参数更合适
// 示例:用 concept 实现 DIP,不依赖虚函数 templateconcept Writable = requires(T t, std::string s) { { t.write(s) } -> std::same_as ; }; template
void log_message(Writer& w, const std::string& msg) { w.write("[LOG] " + msg); }
C++ 的 SOLID 不是贴标签的 checklist,而是要在值语义、移动语义、编译期多态和手动资源控制的夹缝里,找到那个既满足原则又不牺牲性能与安全的平衡点。最容易被忽略的,是把 Java 风格的继承树直接搬进 C++,然后在析构函数里 catch 异常、在接口里塞一堆虚函数、用 std::shared_ptr 包裹一切来“模拟”灵活性——这些都会让 SOLID 变成技术债加速器。










