[设计模式] 浅谈SOLID设计原则

2023-09-14 22:45:12


SOLID是一个缩写词,代表以下五种设计原则

  1. 单一职责原则 Single Responsibility Principle, SRP
  2. 开闭原则 Open-Closed Principle, OCP
  3. 里氏替换原则 Liskov Substitution Principle, LSP
  4. 接口隔离原则 Interface Segregation Principle, ISP
  5. 依赖倒转原则 Dependency Inversion Principle, DIP

注:
这些原则是Robert C. Martin 于21世纪初提出的,实际上这只是从其著作&博客中几十条中选择出来的5条。Robert C. Martin,世界级编程大师,设计模式和敏捷开发先驱,敏捷联盟首任主席,C++ Report前主编。20世纪70年代初成为职业程序员,后创办Object Mentor公司并任总裁。后辈程序员亲切地称之为“Bob大叔”。

单一职责原则

现在有一个日记类Journal,包含标题和若干条记录,通过代码可以描述为:

class Journal
{
	std::string title;
	std::vector<std::string> entries;

	explicit Journal(const std::string& _title) : title{_title}{}
};

现在,添加一个 增加记录 的功能,每条记录以一个按序号增加的数字为前缀:

void Journal::add(const std::string& entry)
{
	static uint32_t count = 1;
	entries.push_back(to_string(count) + ": " + entry);
}

现在对于这个类的使用,我们可以直接:

Journal j{ "Happy Day" };

j.add("This is first log");
j.add("This is second log");

在这里插入图片描述
将 “添加记录”功能作为Journal类的一部分从业务角度来说,是有意义的,因为日记类本身就需要这个功能,维护日记中的记录是Journal类的职责,所以说Journal类中任何与记录有关的功能都是合理的。

现在有了一个新的需求:将日记信息持久化(简单来说保存到文件吧),那么也可以在Journal中添加一块代码:
void Journal::save(const std::string& filePath)
{
	std::ofstream ofs(filePath);
	for (const auto& s : entries)
	{
		ofs << s << endl;
	}
}

代码好写,但是,这么做是正确的吗?(不能够说这种方式是错的,但至少是不正确的),因为: Journal类的职责是维护日志记录,而不是将他们写到磁盘持久化!
如果将这个功能添加到有持久化需求的类中,有关持久化方法的改动都需要在每个类中修改,这显然不是我们想要的!
比较好的方法是:单独考虑这个功能,将这个feature添加到独立的类中,比如可以:

class PersistenctManager
{
	static void save(const Journal& j, const std::string& filePath)
	{
		std::ofstream ofs(filePath);
		for (const auto& s : j.entries)
		{
			ofs << s << endl;
		}
	}
};

这样就比前面的方式好了很多,涉及到持久化的地方只需要改动一处,而不需要往每个涉及到的类中去修改
这正是所说的单一职责原则:每个类只有一个职责,因此也只有一个修改该类的原因(当然是只有该职责发生变化时)

开闭原则

里氏替换原则

接口隔离原则

依赖倒转原则

更多推荐

线程的方法(未完成)

线程的方法1.获取当前线程对象:CurrentThread()@Testpublicvoidbb(){Threadthread=Thread.currentThread();System.out.println(thread.getName());//打印main}2.currentThread().setName和c

CPP-Templates-2nd--第 21 章 模板和继承

目录21.1空基类优化(TheEmptyBaseClassOptimization,EBCO)21.1.1布局原则21.1.2将数据成员实现为基类21.2TheCuriouslyRecurringTemplatePattern(CRTP)21.2.1TheBarton-NackmanTrick21.2.2运算符的实现(

客户端和服务端信息交互模型

什么是客户端和服务端?客户端:可以向服务器发请求,并接收返回的内容进行处理服务器端:能够接收客户端请求,并且把相关资源信息返回给客户端的当用户在地址栏中输入网址,到最后看到页面,中间都经历了什么?后面会详细解析每个步骤干的事一、URL地址解析A:URI/URL/URNURI(UniformResourceldentif

云服务部署:AWS、Azure和GCP比较

🌷🍁博主猫头虎(🐅🐾)带您GotoNewWorld✨🍁🦄博客首页——🐅🐾猫头虎的博客🎐🐳《面试题大全专栏》🦕文章图文并茂🦖生动形象🐅简单易学!欢迎大家来踩踩~🌺🌊《IDEA开发秘籍专栏》🐾学会IDEA常用操作,工作效率翻倍~💐🌊《100天精通Golang(基础入门篇)》🐅学会Gol

【腾讯云 Cloud Studio 实战训练营】提升开发效率与协作:探索腾讯云 Cloud Studio 的强大功能与优势

文章目录一、前言二、认识腾讯云CloudStudio2.1什么是云端开发环境2.2CDE的特点与优点2.2.1提高效率,开发环境一键运行2.2.2提高生产力,可以并行的工作2.2.3开发更加规范2.2.4提供监管,降低资本与资源2.3云端开发环境具备的四大要素2.4腾讯云CloudStudio强大的AI功能2.4.1与

tauri vue vite elemrntui

tauri+vue+viterust```根据https://www.rust-lang.org/tools/install,安装rust。如果是windows会跳出vs工具的安装器,会自动勾选要安装的,直接点安装即可执行cargo--version检查安装是否完成,可以使用cargo创建一个hello项目验证rust

OpenCV中的HoughLines函数和HoughLinesP函数到底有什么区别?

一、简述基于OpenCV进行直线检测可以使用HoughLines和HoughLinesP函数完成的。这两个函数之间的唯一区别在于,第一个函数使用标准霍夫变换,第二个函数使用概率霍夫变换(因此名称为P)。概率版本之所以如此,是因为它仅分析点的子集并估计这些点都属于同一条线的概率。此实现是标准霍夫变换的优化版本,在这种情况

Go语言实践案例之简单字典

一、程序要实现效果:在命令行调用程序的时候,可以在命令行的后面查询一个单词,然后会输出单词的音标和注释。二、思路分析:定义一个结构体DictRequest,用于表示翻译请求的数据结构。其中包含了TransType(翻译类型)、Source(源语言单词)、UserID(用户ID)等字段。定义一个结构体DictRespon

GO编程实践:如何高效使用变量

GO语言是一种强类型、静态编译的编程语言,它具有简洁的语法和强大的并发支持。在GO语言中,变量的定义和使用是非常重要的基本概念之一。下面是关于如何在GO语言中定义变量的详细说明,使用Markdown格式呈现:GO语言变量定义在GO语言中,变量的定义涉及到两个关键步骤:声明和初始化。首先,我们需要声明变量的类型,然后可以

MyBatis核心配置文件解析: 一步步深入理解mybatis-config.xml

😀前言在进行MyBatis项目开发时,合理和高效的配置是确保项目顺利进行的基础。其中,mybatis-config.xml配置文件扮演着极其重要的角色,它包含了MyBatis运行时的各种必要配置信息,如数据库连接属性、事务管理器配置、别名配置等。.提供了一份详细的mybatis-config.xml配置文件解析,一步

Go语言开发环境搭建指南:快速上手构建高效的Go开发环境

Go官网:https://go.dev/dl/Go语言中文网:https://studygolang.com/dl下载Go的语言包进入官方网站Go官网或Go语言中文网:选择下载对应操作系统的安装包:等待下载完成:安装Go的语言包双击运行上一步下载好的Go语言包,点击【Next】:勾选【Iacceptthetermsin

热文推荐