各位看官大大们,晚上好。好久不见,我想死你们了...

先说说写这个系列文章的背景:

工作了这么久了,每天都忙着写业务,好久没有好好静下心来好好总结总结了。正好这段时间公司组织设计模式的分享分,所以我才有机会在这里和大家唠唠嗑。

也许因为自己是小白自学的吧,所以磕磕绊绊走了好多弯路。所以我深刻的理解到在自学的时候有一个前辈在前面引路是多么重要。可以让你少走很多弯路。

拥有更高的学习效率。特别是在一些问题上,苦思冥想很久都没有结果,白白浪费了很多时间,也许有人点拨一下就能茅塞顿开。

有这么一句话说的:如果你想真正掌握某个知识,那就尝试把它教给别人吧。所以我苦思冥想了很久终于决定写一个系列的文章,是为了帮助别人也是为了提升自己。

好了话不多说,下面咋们上干货。

说道设计模式呢,我相信很多人都学习过。但是设计模式到底是什么呢?我相信每个人心中都有自己的答案,就像一千个人心中有一千个哈姆雷特。

那么他到底是什么呢?从字面意思来说,它就是代码设计时解决某些问题的套路。(哎...自古真情留不住唯有套路得人心啊。)

好了,看到这看官们就要问了,你这说的啥玩意,干货呢?尽在这扯玄学,这玩意到底应该怎么学习?

各位看官莫慌,下面就带大家进入设计模式的海洋。设计模式多不胜数,常见的设计模式就有23种(工作中我也只用到了代理模式,策略模式,单例模式,工厂模式和建造者模式)。

在学习他们之前,我们先要了解这些模式的设计思想,所谓知其然还要知其所以然。下面要讲到的就是设计模式的六大原则之单一职责原则。

单一职责原则(Single Responsibility Principle)简称SRP:

定义:There should never be more than one reason for a class to change(原文解释)

中文解释:应该有且仅有一个原因引起类或者方法的变更

单一职责原则的好处:

  1. 类的复杂性降低,实现什么职责都有清晰明确的定义。
  2. 复杂性降低也降低了出现问题的概率,并且提高了可读性和可维护性。
  3. 变更引起的风险降低,变更是不可避免的,如果接口的单一职责做得好,一个接口的修改只对相应的实现类有影响,对其他的接口无影响,这对系统的扩展性、维护性都有非常大的帮助。

看到这的看官是不是准备开骂了。这他妈什么玩意,讲了半天,还是照本宣科。搞个锤子!

各位看官不要急,请往下看。

先请各位看一个类图:

各位看官大大有没有发现什么异常?

现在我来揭晓答案:

这是举例的是一个我工作中遇到的反面例子,这个抽象类和接口很明显的违反了单一职责原则。它包含了数据采集,解析,转换,清洗,映射,计算,保存等多种职责。

期初是为了方便开发,大家便绞尽脑汁封装了一个公共处理类供大家使用。开始项目初期数据来源少,并没有发生什么问题,大家也很满意。

但是(大家注意转折来了)随着项目的推进,数据来源、数据类型越来越多,随之而来的数据映射方式也越来越多。

为了适应各种映射方式大家开始在公共类中加入一些可以覆写的空方法,用子类继承公共类覆写某些方法来实现改变(这里使用的是模板设计模式)。

这也导致这个公共基类被频繁修改经常出现各种莫名其妙的bug,很多之前好的功能也会突然出现问题。而且后来方法越加越多,原先的方法也越来越长逻辑越来越难看懂,导致后面甚至都无法维护,

大家焦头烂额,最后只能推到从来重新划分职责细化方法。最后我们花费了大量的人力物力将之前的抽象类改造成了如下所示五个类:

AbstractDataCapture类:职责是数据采集 由它专门负责数据采集,该类包含主要抽象方法是数据采集方法(由具体子类自己实现)和一些辅助方法。

AbstractDataParser类:职责是数据解析,它专门负责数据解析,该类包含主要抽象方法是数据解析方法(由具体子类自己实现)和一些辅助方法。

AbstractDataCleansing类:职责是数据清洗,它专门负责数据转换成映射model类和清洗数据,该类包含主要抽象方法是数据转换方法和数据清洗方法(由具体子类自己实现)以及一些辅助方法。

AbstractDataHandler类:职责是数据处理,它专门负责数据处理,该类包含主要抽象方法是数据关系映射和数据的计算处理(由具体子类自己实现)以及一些辅助方法。

AbstractDataSave类:职责是数据持久化,该类包含主要方法是数据持久化方法(非抽象方法有公共实现)和一些辅助方法。

这是我实际经历的事情,这是一次血的教训。

 总结:

大家看了之后是不是像反思一下了,我以前的设计是不是有问题了?不,并不是这样的,不要怀疑自己。

原则本身是非常优秀的,但是现实有现实的难处,因为职责和变化因数难以被具体量化。

而且在日常开发中,我们还要考虑工期,成本,人员技术水平等等多种情况。原则是死的人事活的。

我们应该审视夺度,在适当的时候用适当的设计。我个人的意见是接口和方法设计一定要做到职责单一,类的设计尽量做到。

深入浅出系列第一篇(设计模式之单一职责原则)——从纯小白到Java开发的坎坷经历的更多相关文章

  1. 北风设计模式课程---单一职责原则(Single Responsibility Principle)

    北风设计模式课程---单一职责原则(Single Responsibility Principle) 一.总结 一句话总结: 一个类应该有且只有一个变化的原因:单一职责原则(SRP:Single Re ...

  2. 设计模式之单一职责原则(SRP)

    自己之前写过一些关于设计模式的博客,但是大部分都写得比较匆忙.现在正好趁年前有时间,笔者打算好好地整理一下自己这块知识结构.开篇的第一个原则就是设计原则里面最简单的一个原则--单一职责原则. 想必大家 ...

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

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

  4. 设计模式之单一职责原则(iOS开发,代码用Objective-C展示)

    单一职责原则:就一个类而言,应该只有一个引起它变化的原因. 在iOS开发中,我们会很自然的给一个类添加各种各样的功能,比如随便写一个简单的应用程序,一般都会生成一个viewController类,于是 ...

  5. JAVA设计模式之单一职责原则

    概念: 就一个类而言应该只有一个因其他变化的原因. 流程: 问题由来:设类或接口类C负责两个不同不同的职责:职责T1,职责T2.当由于职责T1需求改变进而需要修改类C时,可能导致职责T2收到不可预知的 ...

  6. C#设计模式系列:单一职责原则(Single Responsibility Principle)

    1.单一职责原则的核心思想 一个类应该有且只有一个变化的原因. 2.为什么要引入单一职责原则 单一职责原则将不同的职责分离到单独的类,每一个职责都是一个变化的中心.当需求变化时,这个变化将通过更改职责 ...

  7. C#软件设计——小话设计模式原则之:单一职责原则SRP

    前言:上篇C#软件设计——小话设计模式原则之:依赖倒置原则DIP简单介绍了下依赖倒置的由来以及使用,中间插了两篇WebApi的文章,这篇还是回归正题,继续来写写设计模式另一个重要的原则:单一职责原则. ...

  8. java设计模式学习笔记--单一职责原则

    单一职责原则注意事项和细节 1.降低类的复杂度,一个类只负责一项职责 2.提高可读性,可维护性 3.降低变更引起的风险 4.通常情况下,我们应当遵守单一职责原则,只有逻辑足够简单,才可以在代码级违反单 ...

  9. 深入学习jQuery选择器系列第一篇——基础选择器和层级选择器

    × 目录 [1]id选择器 [2]元素选择器 [3]类选择器[4]通配选择器[5]群组选择器[6]后代选择器[7]兄弟选择器 前面的话 选择器是jQuery的根基,在jQuery中,对事件处理.遍历D ...

随机推荐

  1. igate(因特网网关)

    网关:Gateway 又称网间连接器.协议转换器.-----复杂的网络互连设备. 网关在网络层以上实现网络互连,是复杂的网络互连设备,仅用于两个高层协议不同的网络互连.网关既可以用于广域网互连,也可以 ...

  2. 数据解析_xpath

    重点推荐这种解析方式,xpath是最常用且最便捷高效的一种解析方式,通用性 1.解析原理 1.实例化一个etree的对象,且需要将被解析的页面源码数据加载到改对象中. 2.调用etree对象中的xpa ...

  3. ORA-39257: Data cannot be remapped for more than 10 columns.

    ORA-39257: Data cannot be remapped for more than 10 columns. 前言 还是脱敏数据相关的事情. 使用expdp的remap_data参数对指定 ...

  4. P3574 FAR-FarmCraft 题解

    题目 In a village called Byteville, there are \(n\) houses connected with \(n-1\) roads. For each pair ...

  5. plus.runtime.quit()是个好函数

    对于H5+APP开发,应用的生命周期监听函数里是对应用行为的监控,但是并不对应用执行退出或重启操作.相关操作还是要使用mui

  6. 初识C#扩展方法

    1)扩展方法是什么? 扩展方法可以在不修改原有类的代码前提下,给类“增加”一个方法.扩展方法虽然属于静态方法,但调用的语法却和对象调用类似.直接用一个例子来演示扩展方法. 1.准备实体类 public ...

  7. Qt-文件系统

    1  简介 参考视频:https://www.bilibili.com/video/BV1XW411x7NU?p=45 参考文档:<Qt教程.docx> 说明:本文主要介绍Qt的文件系统. ...

  8. JVM 专题十三:运行时数据区(八)直接内存

    1. 直接内存 不是虚拟机运行时数据区的一部分,也不是<Java虚拟机规范>中定义的内存区域. 直接内存是Java堆外的.直接向系统申请的内存区间. 来源于NIO,通过存在堆中的Direc ...

  9. MYSQL 之 JDBC(十): JDBC的元数据

    可以从Connection对象中获得有关数据库管理系统的各种信息 获取这些信息的方法都是在DatabaseMetaData类中. DatabaseMetaData:描述数据库的元数据对象 Result ...

  10. 01 flask源码剖析之werkzurg 了解wsgi

    01 werkzurg了解wsgi 目录 01 werkzurg了解wsgi 1. wsgi 2. flask之werkzurg源码流程 3. 总结 1. wsgi django和flask内部都没有 ...