寂然解读设计模式 - 单一职责原则

  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方法中,出现了即有森林里的动物,又有天空上的动物,违反了单一职责原则


image-20200827175457704.png

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

我们可以按单一职责原则,将原来的类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 + "正在天空上愉快的飞翔");
    }
}

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

  • 这种修改方法没有对原来的类做大的修改,只是增加了方法

  • 客户端改动范围很小的同时保证了业务逻辑的正确

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

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

通过上面两种方案,大家可以看到,方案一,类级别遵守了单一职责原则,但是改动的代价很高,方案二方法级别遵守了单一职责原则,改动幅度较小,综上所述,单一职责原则最核心的其实就是各司其职

注意事项&细节

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

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

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

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

  • 降低变更引起的风险

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

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

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

如何遵守单一职责原则?

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

其实就是合理的职责分解,相同的职责放到一起,不同的职责分解到不同的接口和实现中去,这个是最容易也是最难运用的原则,关键还是要从业务出发,从需求出发,识别出同一种类型的职责

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

下节预告

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

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,530评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 86,403评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,120评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,770评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,758评论 5 367
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,649评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,021评论 3 398
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,675评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,931评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,659评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,751评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,410评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,004评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,969评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,203评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,042评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,493评论 2 343