【iOS-设计模式】六大设计原则之单一职责原则(SRP,Single Responsibility Principle)

定义

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

定义解读

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

优点

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

问题提出

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

解决方案

遵循单一职责原则。分别建立两个类 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。所以记住,在职责扩散到我们无法控制的程度之前,立刻对代码进行重构。)

示例

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

代码如下所示:
.h

@interface Employee : NSObject
// 计算工资
- (void)calculateSalary:(NSString *)name;
@end

.m

#import "Employee.h"
@implementation Employee
- (void)calculateSalary:(NSString *)name
{
    NSLog(@"%@的工资是:100",name);
}
@end
调用代码
Employee *employee = [[Employee alloc] init];
[employee calculateSalary:@"张三"];
[employee calculateSalary:@"李四"];
输出结果
张三的工资是:100

李四的工资是:100

产品上线后,问题出来了,因为员工的岗位不同,工资的计算是不一样的。修改时如果遵循单一职责原则,需要将 Employee 类细分为总监类 Director、经理类 Manager、普通员工类 Staff,这三个类的实现代码和 Employee 类一样,只是方法 calculateSalary 有所不同,调用代码如下:

Director *director = [[Director alloc] init];
Manager *manager = [[Manager alloc] init];
Staff *staff = [[Staff alloc] init];

[director calculateSalary:@"张三"];
[manager calculateSalary:@"李四"];
[staff calculateSalary:@"王五"];
输出结果
张三总监的工资是:10000

李四经理的工资是:1000

王五员工的工资是:100

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

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

// 计算工资,增加员工岗位的标识(Director:总监;Manager:经理;Staff:普通员工)
- (void)calculateSalary:(NSString *)name flag:(NSString *)flag
{
    if ([flag isEqualToString:@"Director"])
    {
        NSLog(@"%@总监的工资是:10000",name);
    }
    else if ([flag isEqualToString:@"Manager"])
    {
        NSLog(@"%@经理的工资是:1000",name);
    }
    else if ([flag isEqualToString:@"Staff"])
    {
        NSLog(@"%@员工的工资是:100",name);
    }
}
调用代码
Employee *employee = [[Employee alloc] init];
[employee calculateSalary:@"张三" flag:@"Director"];
[employee calculateSalary:@"李四" flag:@"Manager"];
[employee calculateSalary:@"王五" flag:@"Staff"];
输出结果
张三总监的工资是:10000

李四经理的工资是:1000

王五员工的工资是:100

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

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

// 总监工资计算
- (void)directorCalculateSalary:(NSString *)name;
// 经理工资计算
- (void)managerCalculateSalary:(NSString *)name;
// 普通员工工资计算
- (void)staffCalculateSalary:(NSString *)name;
调用代码
Employee *employee = [[Employee alloc] init];
[employee directorCalculateSalary:@"张三"];
[employee managerCalculateSalary:@"李四"];
[employee staffCalculateSalary:@"王五"];
输出结果
张三总监的工资是:10000

李四经理的工资是:1000

王五员工的工资是:100

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

示例总结

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

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

推荐阅读更多精彩内容