01-单一职责原则(SPR)
1. 背景
类T负责两个不同的职责:职责P1,职责P2。当由于职责P1需求发生改变而需要修改类T时,有可能会导致原本运行正常的职责P2功能发生故障。
2. 定义
不要存在多于一个导致类变更的原因。通俗的说,即一个类只负责一项职责。
3. 宏观上
类层次上存在单一职责原则,同样方法层次上也存在单一职责原则。
4. 补充两个概念
高内聚:内聚是指类内部的属性和行为,高内聚就是指:一个类的属性和行为与这个类非常密切,称为高内聚。
低耦合:耦合是指类与类之间或者模块与模块之间的联系,低耦合就是指:耦合度低,易重用、使用灵活。
5. 案例
需求一:有牛和羊两种动物,它们都需要呼吸空气。
解决思路:构造动物类,封装呼气空气的方法。
public class Animal
{
public string myName { get; set; } private string _name = null;
public Animal()
{ }
public Animal(string name)
{
this._name = name;
} //封装呼吸空气的方法
public void BreathAir()
{
Console.WriteLine(this._name + "呼吸空气");
}
}
需求变更:又多了鱼这个物种,它们需要呼吸空气。
解决方案一:直接在BreathAir上加判断,但是这样违背了方法意义上的单一职责原则。
public void BreathAir()
{
if (this._name.Equals("鱼"))
{
Console.WriteLine(this._name + "呼吸水"); //违背了方法级别的单一职责
}
else
{
Console.WriteLine(this._name + "呼吸空气");
}
}
解决方案二:在Animal类中添加一个BreathWater的方法,但是这样违背了类意义上的单一职责原则。
//封装呼吸水的方法
public void BreathWater()
{
Console.WriteLine(this._name + "呼吸水");
}
解决方案三:新建一个WaterAnimal类,在里面封装BreathWater的方法,无论是类级别还是方法级别,都满足了单一职责原则,但是改动成本太大,耗时太长。
public class WaterAnimal
{
private string _name = null;
public WaterAnimal(string name)
{
this._name = name;
}
//封装呼吸水的方法
public void BreathWater()
{
Console.WriteLine(this._name + "呼吸水");
}
}
综合调用:
public static void show()
{
//下面模拟单一职责面对需求变更的情况。
//需求1:有牛和羊两种动物,它们需要呼吸空气
//解决思路:构造动物类→封装呼吸空气的方法
Animal animal1 = new Animal("牛");
Animal animal2 = new Animal("羊");
animal1.BreathAir();
animal2.BreathAir(); //需求2:又多了鱼这个物种,它们需要呼吸水
//解决方案一:在Breath里加判断,但是违背了方法级别的单一职责原则
Animal animal3 = new Animal("鱼");
animal3.BreathAir(); //解决方案二:在Animal类中添加一个BreathWater的方法,但是违背了类级别的单一职责原则
Animal animal4 = new Animal("鱼");
animal4.BreathWater(); //解决方案三:新建一个WaterAnimal类,在里面封装BreathWater的方法,无论是类级别还是方法级别,都满足了单一职责原则,但是改动成本太大,耗时太长
WaterAnimal animal5 = new WaterAnimal("鱼");
animal5.BreathWater(); }
解决方案四:利用委托来解决。
/// <summary>
/// 利用委托的插件式编程
/// </summary>
public static void show2()
{
Console.WriteLine("------------------下面是利用委托来解决--------------------");
Animal animal1 = new Animal() { myName = "牛" };
Animal animal2 = new Animal() { myName = "羊" };
Animal animal3 = new Animal() { myName = "鱼" };
Action<Animal> act1 = Utils.BreathWater;
Action<Animal> act2 = Utils.BreathAir;
act1.Invoke(animal1);
act1.Invoke(animal2);
act2.Invoke(animal3);
}
6. 单一职责原则的好处
(1). 降低类复杂性,一个类只负责一项职责要比负责多项职责简单的多。
(2). 提高类的可读性,便于维护。
(3). 变更引起的风险低,修改一个类的功能,显著降低对其他类的影响。
7. 我的总结
如果完全遵循单一职责原则,当业务复杂的时候,改动成本是最大的。所以我的原则是:根据实际业务场景,来决定是否违背方法或类层次上的单一职责原则。
01-单一职责原则(SPR)的更多相关文章
- 080 01 Android 零基础入门 02 Java面向对象 01 Java面向对象基础 01 初识面向对象 05 单一职责原则
080 01 Android 零基础入门 02 Java面向对象 01 Java面向对象基础 01 初识面向对象 05 单一职责原则 本文知识点:单一职责原则 说明:因为时间紧张,本人写博客过程中只是 ...
- C#设计模式系列:单一职责原则(Single Responsibility Principle)
1.单一职责原则的核心思想 一个类应该有且只有一个变化的原因. 2.为什么要引入单一职责原则 单一职责原则将不同的职责分离到单独的类,每一个职责都是一个变化的中心.当需求变化时,这个变化将通过更改职责 ...
- C#软件设计——小话设计模式原则之:单一职责原则SRP
前言:上篇C#软件设计——小话设计模式原则之:依赖倒置原则DIP简单介绍了下依赖倒置的由来以及使用,中间插了两篇WebApi的文章,这篇还是回归正题,继续来写写设计模式另一个重要的原则:单一职责原则. ...
- 单一职责原则(Single Responsibility Principle)
单一职责原则(SRP:The Single Responsibility Principle) 一个类应该有且只有一个变化的原因. There should never be more than on ...
- SRC单一职责原则
一.定义 一个类应该只有一个发生变化的原因. 二.为什么要使用SRC 因为每一个职责都是变化的一个轴线.当需求变化时,这种变化就会反映为类的职责的变化.如果一个类承担了多于一个的职责,那么引起它变化的 ...
- 开放-封闭原则(OCP)开-闭原则 和 依赖倒转原则,单一职责原则
单一职责原则 1.单一职责原则(SRP),就一个类而言,应该仅有一个引起它变化的原因 2.如果一个类承担的职责过多,就等于把这些职责耦合在一起,一个职责的变化可能会消弱或抑制这个类完成其他职责的能力. ...
- 敏捷软件开发:原则、模式与实践——第8章 SRP:单一职责原则
第8章 SRP:单一职责原则 一个类应该只有一个发生变化的原因. 8.1 定义职责 在SRP中我们把职责定义为变化的原因.如果你想到多于一个的动机去改变一个类,那么这个类就具有多于一个的职责.同时,我 ...
- 第2章 面向对象的设计原则(SOLID):1_单一职责原则(SRP)
1. 单一职责原则(Single Responsibility Principle,SRP) 1.1 单一职责的定义 (1)定义:一个类应该仅有一个引起它变化的原因.这里变化的原因就是所说的“职责”. ...
- 1.单一职责原则(Single Responsibility Principle)
1.定义 就一个类而言,应该仅有一个引起它变化的原因. 2.定义解读 这是六大原则中最简单的一种,通俗点说,就是不存在多个原因使得一个类发生变化,也就是一个类只负责一种职责的工作. 3.优点 类的复杂 ...
- 【设计模式六大原则1】单一职责原则(Single Responsibility Principle)
http://blog.csdn.net/zhengzhb/article/category/926691/1 图片素材来源,java学习手册 ps.内容为自己整理 定义:不要存在多于一个 ...
随机推荐
- win8和win7下解决php5.3和5.4、5.5等不能加载php_curl.dll的终极解决办法 收藏
win8和win7下解决php5.3和5.4.5.5等不能加载php_curl.dll的终极解决办法 收藏2015年01月11日 最近分别在WIN7和Windows8 上分别安装php 高版本!都遇到 ...
- 【转帖】intel 2018年1 月2号爆出漏洞分析 知乎匿名用户
作者:匿名用户链接:https://www.zhihu.com/question/265012502/answer/288407097来源:知乎著作权归作者所有.商业转载请联系作者获得授权,非商业转载 ...
- Git 换行符检查 CRLF 与 LF
遇到的问题 在 git 提交或是签出时,提示如下问题: [git] warning: LF will be replaced by CRLF | fatal: CRLF would be replac ...
- Tomcat7/8访问Server Status、Manager App、Host Manager出现403 forbidden
在配置好Tomcat7/8后,我们往往需要访问Tomcat7/8的Manager以及Host Manager.就需要在tomcat-users.xml中配置用户角色来实现.在地址栏输入:localho ...
- jquery 語法
基本形式: $(selector).action() 文檔加載函數: $(document).Ready{ function(){ //將所有的函數寫到文檔加載函數里,可以防止頁面未加載完全,就執行j ...
- python之tkinter使用-单级菜单
# 菜单功能说明:单级菜单 import tkinter as tk root = tk.Tk() root.title('菜单选择') root.geometry('200x60') # 设置窗口大 ...
- 捕捉JDialog的关闭事件
捕捉JDialog的关闭事件 http://xxqn.iteye.com/blog/431190 public class EditJDialog extends javax.swing.JDialo ...
- BZOJ3456 城市规划(多项式求逆)
设f[i]为连通图的数量,g[i]为不连通图的数量,显然有f[i]=2i*(i-1)/2-g[i],g[i]通过枚举1所在连通块大小转移,有g[i]=Σf[j]*C(i-1,j-1)·2(i-j)*( ...
- hdu 3727 Jewel (可持久化线段树+bit)
链接: http://acm.hdu.edu.cn/showproblem.php?pid=3727 题意: 对一段序列进行四种操作: Insert x :在序列尾部插入一个x: Query_1 s ...
- MT【24】一道五次方程的求根题
解答: 评:一般的五次及以上的多项式方程是无根式解的,只能用计算机去精确到某某位.但是特殊的比如$x^5=1$显然有根式解,本题就是一个不平凡的特殊的例子,这里的代换用于求解三次方程的求根过程是一样的 ...