I walk very slowly, but I never walk backwards

设计模式 - 单一职责原则


寂然

大家好,我是寂然,本节课呢,我来给大家介绍设计模式原则之单一职责原则,带领大家揭开设计模式原则的神秘面纱,话不多说,我们进入正题。不知道大家是否遇到过下面这样的情况:


在实际开发的过程中,有时候大家会发现自己写的类越来越大,几百一千行,该类的功能也越来越多,有一些开发者包括之前的我,看到自己写的类够大,功能够多,可能有点小自豪对吧,看,这是朕写下的江山!!但是当某个功能需要做一个小改动时,就会发现整个程序出现了各种大大小小的问题,然后头发越来越少…


为什么只对这个类的一个功能做了小小的修改就会引起这么大的问题?其实就是因为我们违反了单一职责原则,将多种功能集成在一个类中,就等于把这些功能耦合了起来,一个功能的变化可能会削弱或者抑制这个类完成其他职责的能力,而如果想要避免这种现象的发生,就要尽可能的遵守单一职责原则,那什么是单一职责原则呢?

当然,我们还是首先来看一下单一职责原则的定义

官方定义

单一职责原则(Single Responsibility Principle, SRP),有且仅有一个原因引起类的变更

基本介绍

那根据上面给出的定义,我们来对单一职责原则进行一个基本介绍

即对类来说,一个类应该只负责一项职责。如类 A 负责两个不同职责:职责 1,职责 2,当职责 1 需求变更而改变 A 时,可能造成职责 2 执行错误,所以需要将类 A 的粒度分解为 A1,A2


什么意思呢,给大家举个栗子,如果你项目中DAO层的一个类,既操作user表,又操作了order表,也就是说这个javaBean既负责user表的增删改查,又负责order表的增删改查,那么这个类负责了两个不同的职责,就违反了单一职责原则,所以根据原则需要将这个类的粒度分解为一个userDao操作user表,orderDao来操作order表

案例:动物世界

大家听了单一职责原则的介绍,那我们来看如下一个案例
有一个动物类,里面定义一个在森林奔跑的方法,然后我们创建动物的实例,调用方法,方法内执行打印操作

public class SingleDemo {public static void main(String[] args) {Animal animal = new Animal();animal.run("老虎");animal.run("狮子");animal.run("老鹰");}
}
//定义动物类
class Animal{//森林奔跑方法public void run(String animal){System.out.println(animal + "正在森林里愉快的奔跑");}
}

首先这段代码没有语法上的问题,但是执行的时候,大家就可以看到,出现了明显的逻辑错误,老鹰是没办法在森林里奔跑的,换句话说,在run方法中,出现了即有森林里的动物,又有天空上的动物,违反了单一职责原则



解决方案一:拆分类为更小粒度

我们可以按单一职责原则,将原来的类Animal,分成多个类,在当前业务逻辑下,每个类负责不同的职责,所以根据上面的案例,我们将Animal类根据奔跑的位置进行拆分,分解成不同的类即可,代码示例如下

public class SingleDemo {public static void main(String[] args) {ForestAnimal forestAnimal = new ForestAnimal();forestAnimal.run("老虎");forestAnimal.run("狮子");SkyAnimal skyAnimal = new SkyAnimal();skyAnimal.fly("老鹰");}
}
class ForestAnimal{//森林奔跑方法public void run(String animal){System.out.println(animal + "正在森林里愉快的奔跑");}
}
class SkyAnimal{//森林奔跑方法public void fly(String animal){System.out.println(animal + "正在天空上愉快的飞翔");}
}

OK,那这样首先确实遵循了单一职责原则,同时也保证了业务逻辑的正确,但是大家同时考虑,这样做的改动很大,我们不仅要拆分类,同时还要大范围修改客户端(即main方法里的代码也要改动)那我们还可以怎样做呢?

解决方案二:原有类进行修改

那上面提到,使用方案一,不仅要拆分类,同时还要修改客户端里的代码,那可能有人想到了,那我直接在原有类的基础上进行改动呢?下面我们来看代码示例

public class SingleDemo {public static void main(String[] args) {Animal animal = new Animal();animal.runForest("老虎");animal.runForest("狮子");animal.runSky("老鹰");}
}
class Animal {//森林奔跑方法public void runForest(String animal) {System.out.println(animal + "正在森林里愉快的奔跑");}//天空飞翔方法public void runSky(String animal) {System.out.println(animal + "正在天空上愉快的飞翔");}
}

那下面我们针对方案二进行一下分析

  • 这种修改方法没有对原来的类做大的修改,只是增加了方法
  • 客户端改动范围很小的同时保证了业务逻辑的正确

那这时可能有人要问了,那这样的写法,同样将森林和天空的动物耦合在一个类里了啊?


这里大家要注意哈,确是如此,但是,方案二虽然没有在类这个级别上遵守单一职责原则,但是在方法级别上,仍然遵守单一职责原则,即一个方法只负责一项职责

通过上面两种方案,大家可以看到,方案一,类级别遵守了单一职责原则,但是改动的代价很高,方案二方

法级别遵守了单一职责原则,改动幅度较小,综上所述,单一职责原则最核心的其实就是各司其职

注意事项&细节

  • 降低类的复杂度,一个类只负责一项职责

    (一个类的职责少了,相应的复杂度就会降低)

  • 提高类的可读性以及可维护性

    (相应的复杂度降低,代码量就会减少,可读性也就会提高,可维护性自然就提高了)

  • 降低变更引起的风险

    (一个类的职责越多,变更的可能性就更大,变更带来的风险也就越大)

  • 通常情况下,我们应当遵守单一职责原则

    (只有逻辑足够简单,才可以在代码级违反单一职责原则,只有类中方法数量足够少,才可以在方法级别保持单一职责原则,参考方案二)

如何遵守单一职责原则?

上面的注意事项中也提到了,身为设计模式七大原则之一,通常情况下,我们的代码应当遵守单一职责原则,那大家肯定或多或少都有这样的疑问,如何遵守呢?

其实就是合理的职责分解,相同的职责放到一起,不同的职责分解到不同的接口和实现中去,这个是最容易

也是最难运用的原则,关键还是要从业务出发,从需求出发,识别出同一种类型的职责

需要说明的一点是:单一职责原则不只是面向对象编程思想所特有的,只要是模块化的程序设计,都适用单一职责

下节预告

下一节,我们正式进入设计模式原则之接口隔离原则的学习,我会为大家用多个案例分析,来解读设计模式原则之接口隔离原则,以及它的注意事项和细节,希望大家在学习的过程中,能够感觉到设计模式的有趣之处,高效而愉快的学习,那我们下期见~

寂然解读设计模式 - 单一职责原则相关推荐

  1. 设计模式-单一职责原则

    设计模式-单一职责原则 单一职责原则使用的是创建型模式 创建型模式 创建型模式对类进行抽象 重点,创建型模式能够将对象的创建和和对象的使用分离.即使用创建型模式能够使得对象的创建,对象的使用分离.重点 ...

  2. 设计模式-单一职责原则-实践运用

    单一职责原则-概念 1.单一职责原则是最简单的面向对象设计原则,它用于控制类的粒度大小. 2.单一职责原则定义如下: 单一职责原则(Single Responsibility Principle, S ...

  3. Java设计模式-单一职责原则

    单一职责原则 [Single Responsibility Principle]   定义:不要存在多于一个导致类变更的原因.通俗的说,即一个类只负责一项职责. 问题由来:类T负责两个不同的职责:职责 ...

  4. 设计模式-02.经典设计原则-第一节-单一职责原则,开闭原则,里式替换,接口隔离【万字长文系列】

    文章目录 设计模式经典设计原则-第一节 单一职责原则(SRP) 如何理解单一职责原则? 如何判断类的职责是否足够单一? 类的职责是否设计得越单一越好? 开闭原则(OCP) 如何理解"对扩展开 ...

  5. 前端中会用到的设计模式之单一职责原则

    1:设计模式应用不应用,取决于对现在和未来判断后的取舍.没必要用尽量不用! 2.设计模式的目的是  减少复杂度(一个函数中包含的功能个数), 降低耦合度(一个对象与其他对象的关系个数).耦合度不能为0 ...

  6. 学习设计模式 - 六大基本原则之单一职责原则

    设计模式总共有六大基本原则,统称为SOLID (稳定)原则,分别是S-单一职责原则(Single Responsibility Principle), O-开闭原则(Open closed Princ ...

  7. 设计模式原则--单一职责原则

    单一职责原则(SRP) 定义:就一个类而言,应该仅有一个引起它变化的原因 场景: 一个公司有3类员工,分别是 主管,程序员,销售 代码: using System; using System.Coll ...

  8. 北风设计模式课程---单一职责原则

    北风设计模式课程---单一职责原则 一.总结 一句话总结: 视频教程网上一定能找到做好笔记的博客,很大几率都不需要自己做笔记.比如北风设计模式课程,https://www.cnblogs.com/xi ...

  9. 设计模式 之 设计的 六大原则(1)单一职责原则

    由于这些原则性东西 属于概念东西,就不具体以代码描述了.以下是摘自网上和自己的一些理解 首先了解一些 面向对象的特性: 面向对象 有 三大基本特征:封装 ,继承, 多态. 封装: 也就是把客观事物封装 ...

最新文章

  1. 一重量级联盟成立!北大、浙大、上交大、国科大等34校加入
  2. “诺奖风向标”2021拉斯克奖公布:授予mRNA疫苗、光遗传学以及戴维·巴尔的摩...
  3. 更简单的调试Release版本Optimize code的.NET程序集
  4. Linux进程的Uninterruptible sleep(D)状态
  5. 理想的计算机职业作文100,我的理想作文100字(通用30篇)
  6. 为什么个体户贷款那么难
  7. IoTSharp部署教程-Sqlite分表篇
  8. 怎样用C语言实现五子棋,C语言实现五子棋
  9. MyBatis的接口式编程Demo
  10. word文档被锁定无法编辑的解决方法
  11. 微信jssdk在iframe页面失效问题的解决措施
  12. 自己实现LinkedList类
  13. 建网站域名服务器那个好,如何选择好建网站的域名?
  14. 计算机类qq网名,最帅的qq名字
  15. python如何爬有道翻译_Python爬虫爬有道翻译
  16. shopnc mysql_shopnc配置
  17. Cortana 设备建议
  18. 艾永亮:酒瓶中的战争,谁是下一瓶被拿起的葡萄酒
  19. 【问题解决】电脑连接自家WiFi不能进B站而手机能进
  20. std::set用法

热门文章

  1. 【Solidity】函数returns多个值的接收方式
  2. HTML报错:UncaughtTypeError
  3. linux下调整分辨率
  4. VirtualBox调整分辨率
  5. (管用)电脑突然没声了
  6. 郭德纲致郭麒麟诫子书
  7. VoLTE的基本介绍
  8. 全球智能网联汽车出货量预计
  9. Bwsaas多端SAAS平台运营系统源码v1.1.1
  10. CSS小应用(圆形关闭按钮、大于号、可旋转三角箭头)