news 2026/4/27 5:47:25

设计模式基础与SOLID原则

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
设计模式基础与SOLID原则

🏗️ 设计模式基础与SOLID原则

设计模式是软件开发中经过验证的、可复用的解决方案。掌握设计模式,能够让我们的代码更加优雅、可维护、可扩展。

一、什么是设计模式

设计模式(Design Pattern)是一套被反复使用、多数人知晓的、经过分类编目的、代码设计经验的总结。使用设计模式是为了可重用代码、让代码更容易被他人理解、保证代码可靠性。

设计模式的核心思想是:在特定环境下解决软件设计中某类通用问题的方案。它不是可以直接复制粘贴的代码,而是一种解决问题的思路和方法论。

🎯 设计模式的本质价值

价值维度说明
可复用性避免重复造轮子,站在巨人的肩膀上
可维护性遵循统一的模式,代码结构清晰易懂
可扩展性方便添加新功能,减少对现有代码的修改
沟通效率提供统一的术语,团队沟通更高效

二、设计模式的三大分类

根据设计模式的用途,可以将其分为三大类:创建型模式结构型模式行为型模式

2.1 创建型模式(5种)

创建型模式关注对象的创建过程,将对象的创建和使用分离,降低系统的耦合度。

模式名称核心思想适用场景
工厂方法模式定义一个创建对象的接口,让子类决定实例化哪一个类需要根据条件创建不同对象
抽象工厂模式提供一个创建一系列相关或相互依赖对象的接口需要创建产品族
单例模式确保一个类只有一个实例,并提供全局访问点配置管理器、连接池、日志对象
建造者模式将一个复杂对象的构建与它的表示分离创建复杂对象,如配置项较多的对象
原型模式通过复制现有对象来创建新对象创建成本较高的对象

2.2 结构型模式(7种)

结构型模式关注类和对象的组合,通过继承或组合来构建更大的结构。

模式名称核心思想适用场景
适配器模式将一个类的接口转换成客户希望的另一个接口接口不兼容时进行转换
装饰器模式动态地给一个对象添加一些额外的职责不通过继承扩展对象功能
代理模式为其他对象提供一种代理以控制对这个对象的访问远程代理、虚拟代理、保护代理
外观模式为子系统中的一组接口提供一个一致的界面简化复杂系统的使用
桥接模式将抽象部分与实现部分分离,使它们都可以独立变化多维度变化的对象
组合模式将对象组合成树形结构以表示"部分-整体"的层次结构树形结构数据处理
享元模式运用共享技术有效地支持大量细粒度的对象大量相似对象的共享

2.3 行为型模式(11种)

行为型模式关注对象之间的通信、职责划分,以及算法的封装。

模式名称核心思想适用场景
策略模式定义一系列算法,把它们一个个封装起来,并且使它们可相互替换多种算法可互换
模板方法模式定义一个操作中的算法骨架,而将一些步骤延迟到子类中固定流程,具体步骤可变
观察者模式定义对象间一对多的依赖关系,当一个对象状态改变时,所有依赖它的对象都得到通知事件驱动系统
迭代器模式提供一种方法顺序访问一个聚合对象中各个元素,而又不需暴露该对象的内部表示遍历集合
责任链模式使多个对象都有机会处理请求,从而避免请求的发送者和接收者之间的耦合关系多级审批、事件处理链
命令模式将一个请求封装为一个对象,从而使你可用不同的请求对客户进行参数化撤销/重做操作
备忘录模式在不破坏封装性的前提下,捕获一个对象的内部状态,并在该对象之外保存这个状态撤销操作、游戏存档
状态模式允许一个对象在其内部状态改变时改变它的行为状态机实现
访问者模式表示一个作用于某对象结构中的各元素的操作数据结构与操作分离
中介模式用一个中介对象来封装一系列的对象交互对象间复杂交互
解释器模式给定一个语言,定义它的文法的一种表示,并定义一个解释器SQL解析、正则表达式

三、SOLID原则详解

SOLID是面向对象设计的五大基本原则的首字母缩写,由Robert C. Martin(Uncle Bob)提出。这些原则是设计模式的理论基础,理解它们对于正确使用设计模式至关重要。

3.1 单一职责原则(Single Responsibility Principle, SRP)

定义:一个类应该只有一个引起它变化的原因。

核心思想:每个类只负责一项职责,不要将多个职责耦合在一个类中。

// ❌ 违反单一职责原则classUser{public:voidlogin(){/* 登录逻辑 */}voidsaveToDatabase(){/* 数据库存储逻辑 */}voidsendEmail(){/* 发送邮件逻辑 */}};// ✅ 遵循单一职责原则classUser{public:voidlogin(){/* 登录逻辑 */}};classUserRepository{public:voidsave(User user){/* 数据库存储逻辑 */}};classEmailService{public:voidsend(User user){/* 发送邮件逻辑 */}};

实际经验

  • 如果一个类的方法超过10个,或者代码超过300行,就需要考虑是否职责过多
  • 职责的划分粒度需要根据实际项目规模来定,不要过度拆分

3.2 开闭原则(Open-Closed Principle, OCP)

定义:软件实体应该对扩展开放,对修改关闭。

核心思想:在不修改现有代码的情况下,通过扩展来增加新功能。

// ✅ 通过抽象实现开闭原则classShape{public:virtualdoublearea()=0;virtual~Shape()=default;};classRectangle:publicShape{doublewidth,height;public:doublearea()override{returnwidth*height;}};classCircle:publicShape{doubleradius;public:doublearea()override{return3.14159*radius*radius;}};// 新增三角形,不需要修改现有代码classTriangle:publicShape{doublebase,height;public:doublearea()override{return0.5*base*height;}};

实际经验

  • 使用抽象和多态是实现开闭原则的主要手段
  • 设计时要预见可能的变化点,提前做好抽象

3.3 里氏替换原则(Liskov Substitution Principle, LSP)

定义:所有引用基类的地方必须能透明地使用其子类的对象。

核心思想:子类必须能够替换其基类,且不破坏程序的正确性。使用多态、面向接口编程。

// ❌ 违反里氏替换原则classBird{public:virtualvoidfly(){/* 飞行 */}};classPenguin:publicBird{public:voidfly()override{throwstd::runtime_error("Penguins can't fly!");}};// ✅ 正确的设计classBird{/* 基类不包含fly */};classFlyingBird:publicBird{public:virtualvoidfly()=0;};classSparrow:publicFlyingBird{/* 可以飞 */};classPenguin:publicBird{/* 企鹅只是鸟,不会飞 */};

实际经验

  • 子类不应该覆盖父类的非抽象方法
  • 子类可以有更严格的前置条件,但必须有更宽松的后置条件

3.4 接口隔离原则(Interface Segregation Principle, ISP)

定义:客户端不应该依赖它不需要的接口。

核心思想:提供尽可能小的单独接口,而不要提供大的总接口。将一个大接口分解为多个小接口。

// ❌ 违反接口隔离原则classIWorker{public:virtualvoidwork()=0;virtualvoideat()=0;};// ✅ 遵循接口隔离原则classIWorkable{public:virtualvoidwork()=0;};classIFeedable{public:virtualvoideat()=0;};classWorker:publicIWorkable,publicIFeedable{// 实现工作接口voidwork()override{/* ... */}// 实现进食接口voideat()override{/* ... */}};classRobot:publicIWorkable{// 机器人只需要工作,不需要进食voidwork()override{/* ... */}};

实际经验

  • 接口的粒度要适中,太大会导致实现类被迫实现不需要的方法,太小会导致接口数量爆炸
  • 一个接口只服务于一个子模块或业务逻辑

3.5 依赖倒置原则(Dependency Inversion Principle, DIP)

定义:高层模块不应该依赖低层模块,两者都应该依赖其抽象;抽象不应该依赖细节,细节应该依赖抽象。

核心思想:面向接口编程。当模块A需要用到模块B的功能时,应当通过一个接口来使用B,而不是直接使用B的实现。

// ❌ 违反依赖倒置原则classMySQLDatabase{public:voidsave(string data){/* MySQL存储 */}};classUserService{MySQLDatabase*db;// 直接依赖具体实现public:voidsaveUser(string data){db->save(data);}};// ✅ 遵循依赖倒置原则classIDatabase{public:virtualvoidsave(string data)=0;virtual~IDatabase()=default;};classMySQLDatabase:publicIDatabase{public:voidsave(string data)override{/* MySQL存储 */}};classPostgreSQLDatabase:publicIDatabase{public:voidsave(string data)override{/* PostgreSQL存储 */}};classUserService{IDatabase*db;// 依赖抽象接口public:UserService(IDatabase*database):db(database){}voidsaveUser(string data){db->save(data);}};

实际经验

  • 依赖注入是实现依赖倒置原则的常用手段
  • 通过构造函数、setter方法或接口传递依赖

四、设计模式与SOLID原则的关系

设计模式是SOLID原则的具体应用,每种设计模式都体现了某些SOLID原则。

设计模式体现的SOLID原则
工厂模式依赖倒置原则、开闭原则
单例模式单一职责原则
策略模式开闭原则、单一职责原则
观察者模式开闭原则、依赖倒置原则
适配器模式开闭原则、接口隔离原则
装饰器模式开闭原则、单一职责原则
模板方法模式开闭原则

💡 理解要点

  1. 设计模式是手段,原则是目的:设计模式是实现设计原则的具体方法
  2. 灵活应用:不要为了使用设计模式而使用,要根据实际需求选择
  3. 权衡取舍:设计模式会增加代码复杂度,需要权衡利弊

五、学习建议

📚 推荐学习路径

第一步:理解面向对象基础 → 第二步:学习SOLID原则 → 第三步:学习设计模式 ↓ 第四步:项目实战应用 ↓ 第五步:阅读优秀源码

🎯 实践建议

  1. 从小处着手:先在小型项目中尝试应用设计模式
  2. 重构现有代码:在代码重构中应用设计模式效果最佳
  3. 阅读开源项目:学习优秀项目如何使用设计模式
  4. 画UML图:用类图辅助理解设计模式的结构
  5. 写技术博客:输出是最好的学习方式

⚠️ 常见误区

误区正确理解
设计模式越多越好根据实际需求选择,过度设计反而有害
设计生搬硬套理解模式的本质,灵活应用
忽视简单性简单的代码优于复杂的模式
只学用法不学原理理解设计原则才能真正掌握设计模式

六、总结

设计模式是软件开发的智慧结晶,SOLID原则是面向对象设计的指导思想。掌握它们能够:

  • 📈 提高代码质量和可维护性
  • 🔄 增强代码的可扩展性和灵活性
  • 🤝 提升团队协作效率
  • 🎯 培养良好的设计思维

记住:设计模式不是银弹,正确的使用场景和方法才是关键。


参考资料

  • C++各类设计模式及实现详解 - 知乎
  • 面向对象5个基本原则 - CSDN
  • 突破编程_C++_设计模式(策略模式)- CSDN
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/27 5:47:22

Janus-Pro-7B面试准备利器:可视化解读Java八股文中的核心概念

Janus-Pro-7B面试准备利器:可视化解读Java八股文中的核心概念 每次准备Java面试,面对那些绕来绕去的“八股文”问题,你是不是也感觉头大?什么JVM内存模型、HashMap底层原理,光看文字描述,脑子里总是一团浆…

作者头像 李华
网站建设 2026/4/27 5:47:21

算法训练营第14天|18. 四数之和

题目链接: https://leetcode.cn/problems/4sum/ 视频链接: https://www.bilibili.com/video/BV1DS4y147US 我的代码: https://leetcode.cn/problems/4sum/submissions/721320141 看到题目的第一想法: 看到四数之和&#xff0…

作者头像 李华
网站建设 2026/4/27 5:44:46

对于点火线圈分配式同时点火的初步理解

点火线圈分配式同时点火,是无分电器点火线圈分配式点火方式中的一种,常用于‌偶数缸发动机‌,如4缸、6缸。该方式和二极管分配式无分电器点火方式有相似的地方,可以和“对于二极管分配式无分电器点火的理解”对比着看。 1、主要特…

作者头像 李华
网站建设 2026/4/27 5:43:40

时间序列预测中的残差建模技术与Python实战

## 1. 时间序列预测中的残差建模核心价值去年帮某电商平台优化销售预测系统时,发现他们的LSTM模型在节假日预测总是出现系统性偏差。当我尝试对预测残差进行二次建模后,周误差率直接下降了37%。这让我意识到,残差修正才是提升时间序列预测精度…

作者头像 李华
网站建设 2026/4/27 5:42:29

Sambert多情感语音合成部署教程:一键启动,快速体验AI语音生成

Sambert多情感语音合成部署教程:一键启动,快速体验AI语音生成 1. 引言:为什么选择Sambert语音合成? 在当今数字化时代,语音合成技术已经广泛应用于智能客服、有声读物、虚拟助手等领域。然而,传统语音合成…

作者头像 李华