1.定义

  就一个类而言,应该仅有一个引起它变化的原因。

2.定义解读

  这是六大原则中最简单的一种,通俗点说,就是不存在多个原因使得一个类发生变化,也就是一个类只负责一种职责的工作。

3.优点

  • 类的复杂度降低,一个类只负责一个功能,其逻辑要比负责多项功能简单的多;
  • 类的可读性增强,阅读起来轻松;
  • 可维护性强,一个易读、简单的类自然也容易维护;
  • 变更引起的风险降低,变更是必然的,如果单一职责原则遵守的好,当修改一个功能时,可以显著降低对其他功能的影响。

4.问题提出

  假设有一个类C,它负责两个不同的职责:职责P1和P2。当职责P1需求发生改变而需要修改类C时,有可能会导致原本运行正常的职责P2功能发生故障。

5.解决方案

  遵循单一职责原则。分别建立两个类C1、C2,使C1完成职责P1,C2完成职责P2。这样,当修改类C1时,不会使职责P2发生故障风险;同理,当修改C2时,也不会使职责P1发生故障风险。

  说到这里,大家会觉得这个原则太简单了。稍有经验的程序员,即使没有听说过单一职责原则,在设计软件时也会自觉的遵守这一重要原则。在实际的项目开发中,谁也不希望因为修改了一个功能导致其他的功能发生故障。而避免出现这一问题的方法便是遵循单一职责原则。虽然单一职责原则如此简单,并且被认为是常识,即便是经验丰富的程序员写出的程序,也会有违背这一原则的代码存在。为什么会出现这种现象呢?因为有职责扩散。实际项目中,因为某种原因,职责P被分化为粒度更细的职责P1和P2。

  比如:类C只负责一个职责P,这样设计是符合单一职责原则的。后来由于某种原因,也许是需求变更了,也许是客户提出了新的功能,需要将职责P细分为粒度更细的职责P1,P2,这时如果要使程序遵循单一职责原则,需要将类C也分解为两个类C1和C2,分别负责P1、P2两个职责。但是在程序已经写好的情况下,这样做有时候需要花费更多的工作量。在项目工期紧张的情况下,我们通常会简单的修改类C,用它来负责两个职责,虽然这样做有悖于单一职责原则。(这样做的风险在于职责扩散的不确定性,因为在未来可能会扩散出P1,P2,P3,P4……Pn。所以记住,在职责扩散到我们无法控制的程度之前,立刻对代码进行重构。)

6.示例

  说一个和我们密切相关的场景:员工的工资计算。刚开始的时候,我们会新建一个员工类,在员工类里面有一个计算工资的方法,代码如下所示:

class Employee
{
func calculateSalary(name: String)
{
print("\(name)的工资是100");
}
} //调用
let employee = Employee();
employee.calculateSalary("小明"); //打印:小明的工资是100

  产品上线后,问题出来了,因为员工的岗位不同,工资的计算是不一样的。修改时如果遵循单一职责原则,需要将Employee类细分为总监类Director、经理类Manager、普通员工类Staff,这三个类的实现代码和Employee类一样,只是方法calculateSalary有所不同。实际项目中,可以考虑将Employee定义为协议,Director、Manager、Staff实现该协议,这样以后扩展其他类型的职位增加相应的类即可。

protocol Employee
{
func calculateSalary(name: String);
} //总监
class Director: Employee
{
func calculateSalary(name: String)
{
print("\(name)总监的工资是10000");
}
} //经理
class Manager: Employee
{
func calculateSalary(name: String)
{
print("\(name)经理的工资是1000");
}
} //普通员工
class Staff: Employee
{
func calculateSalary(name: String)
{
print("\(name)员工的工资是100");
}
} //调用
let director = Director();
director.calculateSalary("张三"); //打印:张三总监的工资是10000
let manager = Manager();
manager.calculateSalary("李四"); //打印:李四经理的工资是1000
let staff = Staff();
staff.calculateSalary("王五"); //打印:王五员工的工资是100

  上面的修改方式是在遵循单一职责原则下进行的,从修改中,我们可以看到,这样修改的工作量相对较大,需要新增不同的岗位类,还需要修改调用代码。实际项目开发中,我们可能会采取如下两种方式:

  【方式一】:直接修改Employee类里面的calculateSalary方法,在这里,我们需要对calculateSalary方法增加一个参数,以标识员工的岗位。

  修改后的calculateSalary方法如下所示:

enum EmployeeType
{
case Director;
case Manager;
case Staff;
} class Employee
{
func calculateSalary(name: String, employeeType: EmployeeType)
{
if .Director == employeeType
{
print("\(name)总监的工资是10000");
}
else if .Manager == employeeType
{
print("\(name)经理的工资是1000");
}
else if .Staff == employeeType
{
print("\(name)的工资是100");
}
}
} //调用
let employee = Employee();
employee.calculateSalary("张三", employeeType: .Director); //打印:张三总监的工资是10000
employee.calculateSalary("李四", employeeType: .Manager); //打印:李四经理的工资是1000
employee.calculateSalary("王五", employeeType: .Staff); //打印:王五的工资是100

  从上面可以看到,这种修改方式相对要简单的多,是直接在方法级别上违背了单一职责原则,虽然修改起来最简单,但隐患却也是最大的。假设有一天需要将总监分为财务总监和研发总监,则又需要修改Employee类的calculateSalary方法,而对原有代码的修改会对已有功能带来风险(可能会存在遗漏或者疏忽)。

  【方式二】:在Employee类中新增不同岗位的工资计算方法,.h文件中新加的方法定义如下所示:

class Employee
{
func directorCalculateSalary(name: String)
{ print("\(name)总监的工资是10000");
} func managerCalculateSalary(name: String)
{
print("\(name)经理的工资是1000"); } func staffCalculateSalary(name: String)
{
print("\(name)的工资是100");
}
} //调用
let employee = Employee();
employee.directorCalculateSalary("张三"); //打印:张三总监的工资是10000
employee.managerCalculateSalary("李四"); //打印:李四经理的工资是1000
employee.staffCalculateSalary("王五"); //打印:王五的工资是100

  可以看到,方式二没有改动原来的方法,而是在类中新加了三个方法,这样虽然也违背了单一职责原则,但在方法级别上却是符合单一职责原则,因为它并没有改变原来方法的代码。

7.示例总结

  上面三种方式各有优缺点,那么在实际编程中,该采用哪一种呢?这个问题没有标准答案,需要根据实际情况来确定。

1.单一职责原则(Single Responsibility Principle)的更多相关文章

  1. 面象对象设计原则之一:单一职责原则(Single Responsibility Principle, SRP)

    单一职责原则是最简单的面向对象设计原则,它用于控制类的粒度大小.单一职责原则定义如下:单一职责原则(Single Responsibility Principle, SRP):一个类只负责一个功能领域 ...

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

    定义:不要存在多于一个导致类变更的原因.通俗的说,即一个类只负责一项职责. 问题由来:类T负责两个不同的职责:职责P1,职责P2.当由于职责P1需求发生改变而需要修改类T时,有可能会导致原本运行正常的 ...

  3. 设计模式六大原则(一):单一职责原则(Single Responsibility Principle)

    单一职责(SRP)定义: 不要存在多于一个导致类变更的原因,通俗的说,即一个类只负责一项职责. 问题由来: 类T负责两个不同的职责:职责P1,职责P2.当由于职责P1需求发生改变而需要修改类T时,有可 ...

  4. 01-01.单一职责原则(Single Responsibility)

    1.基本介绍 对于类来说的,就是一个类,应该只负责一项职责(一个类只管一件事). 如类A负责两个不同职责:职责1,职责2. 当职责1需求变更而改变A时,可能造成职责2执行错误,所以需要将类A的粒度分解 ...

  5. 单一职责原则(Simple responsibility pinciple, SRP)

    一个类只负责一个功能领域中的相应职责 未完待续

  6. 单一职责原则(Single Responsibility Principle)

    单一职责原则(SRP:The Single Responsibility Principle) 一个类应该有且只有一个变化的原因. There should never be more than on ...

  7. 【设计模式】单一职责原则(SRP)

    单一职责原则是面向对象原则五大原则中最简单,也是最重要的一个原则, 他的字面定义如下: 单一职责原则(Single Responsibility Principle, SRP): 一个类只负责一个功能 ...

  8. 【面向对象设计原则】之单一职责原则(SRP)

    单一职责原则是面向对象原则五大原则中最简单,也是最重要的一个原则, 他的字面定义如下: 单一职责原则(Single Responsibility Principle, SRP): 一个类只负责一个功能 ...

  9. 2单一职责原则SRP

    一.什么是单一职责原则 单一职责原则(Single Responsibility Principle ): 就一个类而言,应该仅有一个引起它变化的 原因. 二.多功能的山寨手机 山寨手机的功能: 拍照 ...

  10. 面向对象五大原则_1.单一职责原则&2.里氏替换原则

    单一职责原则:Single Responsibility Principle (SRP) 一个类.仅仅有一个引起它变化的原因.应该仅仅有一个职责.每个职责都是变化的一个轴线.假设一个类有一个以上的职责 ...

随机推荐

  1. 【转】Github轻松上手4-常用的git命令

    转自:http://blog.sina.com.cn/s/blog_4b55f6860100zzih.html 附上一些git的常见命令: •    git remote add origin git ...

  2. Python学习2-列表和元组

    Python学习2-列表和元组 标签(空格分隔): 列表 元组 在Python中,最基本的数据结构是序列(sequence).序列中的每个元素被分配一个序号--即元素的位置,也称为索引.索引从0开始. ...

  3. crontab 每月最后一天

    0 8 28-31 * * [ `date -d tomorrow +%e` -eq 1 ] && do-something   我觉得能想到这种方法的,都是经验丰富的人.程序员们,想 ...

  4. 关于浮动-float

    1.存在浏览器兼容问题:js代码 2.对于这种存在浏览器兼容问题的问题,我们可以绕开兼容性问题,先在css样式写好,然后通过该变className 3.学习的博客 https://paran.io/c ...

  5. 10、NFC技术:读写NFC标签中的文本数据

    代码实现过程如下: 读写NFC标签的纯文本数据.java import java.nio.charset.Charset; import java.util.Locale; import androi ...

  6. IRequiresSessionState和IReadOnlySessionState应用上的一些差异

    在调用ashx时,如果需要应用Session,则必须继承接口 IRequiresSessionState,IReadOnlySessionState,但根据字面,可以知道 IRequiresSessi ...

  7. LoadRunner--内存指标介绍

    Threads——线程数当前全部线程数============================================ Available MBytes——物理内存的可用数指计算机上可用于运行 ...

  8. python程序中自启动appium服务

    普通启动Appium服务方法:      打开cmd,运行命令: #>appium -a 127.0.0.1 -p 4723 当程序输出如上图信息的时候,表示appium启动成功,此时便可以运行 ...

  9. SpringMVC + Spring + MyBatis 学习笔记:遭遇order by 排序问题

    系统:WIN8.1 数据库:Oracle 11GR2 开发工具:MyEclipse 8.6 框架:Spring3.2.9.SpringMVC3.2.9.MyBatis3.2.8 用MyBatis写排序 ...

  10. 使用JavaMail API发送邮件

    发送邮件是很常用的功能,注册验证,找回密码,到货通知,欠费提醒等,都可以通过邮件来提醒. Java中发送邮件需要使用javax.mail.jar包,读者可以上网搜索或去官方下载,下载地址为: 下面贴上 ...