设计模式系列12--职责链模式

image

项目做完了,作为老大的你决定叫上项目组的人一起聚个餐犒劳下大家,于是你去财务申请聚餐费用,财务告诉你聚餐费用申请的流程如下:

  • 小于500元,项目经理可以审批
  • 大于500小于1000,需要部门经理审批
  • 大于1000的都需要总经理审批

如果让你来实现这个流程,该如何做呢?常规做法很简单,如下所示:

#import "feeRequest.h"

@implementation feeRequest

-(void)applayForFee:(NSInteger)fee{
    if (fee < 500){
        [self projectHandle];
    
    }else if(fee < 1000 && fee > 500){
        [self depManagerHandle];
    
    }else{
        [self generalManagerHandle];
    }
}

-(void)projectHandle{
    NSLog(@"项目经理同意了费用申请");
}

-(void)depManagerHandle{
    NSLog(@"部门经理同意了费用申请");
}

-(void)generalManagerHandle{
    NSLog(@"总经理同意了费用申请");

}
@end

实现起来很简单吧。但是仔细分析下,上面的写法有如下两个问题

  • 申请费用的流程可能会经常变动。现在是项目经理-->部门经理--->总经理,如果哪天流程倒过来呢?或者加入了新的审批的人呢?上面的代码就无可避免的需要更改

  • 各个审批流程可能会更改。现在的三个审批流程如上所示,但是可能每个经理的审批的金额会有所更改,这个时候也需要更改上面的代码

抽象下上面的流程:客户提出一个请求(申请聚餐费用),会有很多对象(经理)来处理这个请求,每个对象的处理逻辑是不同的,如果对象自己可以处理这个请求,就会处理完成然后把结果返回给客户,如果自己不能处理,就交给其他对象处理。而且希望上面的流程可以灵活变动,处理请求的对象可以随意组合替换,来适应新的业务需求。

要实现上述功能,可以使用职责链模式,下面就来具体看看


定义

使多个对象都有机会处理请求,从而避免请求的发送者和接收者之间的耦合关系。将这 些对象连成一条链,并沿着这条链传递该请求,直到有一个对象处理它为止。

分析下上面的业务场景,客户端提取申请费用的请求,然后项目经理、部门经理、总经理依次处理这个请求,他们组成了一个请求处理链(项目经理-->部门经理--->总经理),客户请求在这个链中传递,直到一个对象处理了这个请求才结束,否则一直向下传递到链的末尾,这正是职责链的功能。

要想让流程的处理灵活多变,可以随意组合和替换,就需要动态构建流程处理的步骤,每个步骤实现一个功能,然后把这些步骤串联起来实现完整的请求处理链。

我们要实现发送者和接受者解耦,那么就需要让提交帮助请求的对象不需要知道谁是最终提供帮助的对象。职责链模式给多个对象处理一个请求的机会,从而解耦发送者和接受者。该请 求沿对象链传递直至其中一个对象处理它,从第一个对象开始,链中收到请求的对象要么亲自处理它,要么转发给链中的下一个候 选者。提交请求的对象并不明确地知道哪一个对象将会处理它,但是最终会有一个隐式对象来保证请求一定会被处理。

由上述分析可知,职责链模式适用于如下几种情况:

  • 有多个的对象可以处理一个请求,但是具体是哪个对象处理该请求则是在运行时刻才确定。
  • 想在不明确指定接收者的情况下,向多个对象中的一个提交请求。
  • 一个请求的处理链可以动态设置

UML结构图及说明

image

代码实现

1、定义抽象类handler

#import <Foundation/Foundation.h>

@interface handler : NSObject
@property(strong,nonatomic)handler *successor;

-(void)handleRequest:(NSInteger)fee;
@end

==================

#import "handler.h"

@implementation handler


-(void)handleRequest:(NSInteger)fee{
    
}

@end

2、实现三个具体的职责链成员

#import "handler.h"

@interface projectManagerHandler : handler

@end

=============================
#import "projectManagerHandler.h"

@implementation projectManagerHandler

-(void)handleRequest:(NSInteger)fee{
    if (fee < 500) {
        NSLog(@"项目经理同意了费用申请");
    }else{
        if (self.successor)
            NSLog(@"项目经理没有权限批准,转到部门经理处理");
            [self.successor handleRequest:fee];
    }
}
@end


#import "handler.h"

@interface depManagerHandler : handler

@end

=====================
#import "depManagerHandler.h"

@implementation depManagerHandler

-(void)handleRequest:(NSInteger)fee{
    if (fee > 500 && fee < 1000) {
        NSLog(@"部门经理同意了费用申请");
    }else{
        if (self.successor)
            NSLog(@"部门经理没有权限批准,转到总经理处理");
            [self.successor handleRequest:fee];
    }
}

@end


#import "handler.h"

@interface generalManagerHandler : handler

@end


=====================

#import "generalManagerHandler.h"

@implementation generalManagerHandler
-(void)handleRequest:(NSInteger)fee{
    NSLog(@"总经理同意了费用申请");
}
@end

3、测试

     handler *handler1 = [projectManagerHandler new];
        handler *handler2 = [depManagerHandler new];
        handler *handler3 = [generalManagerHandler new];
        
        //设置责任链中的下一个处理对象
        handler1.successor = handler2;
        handler2.successor = handler3;
        
        [handler1 handleRequest:100];
        
        NSLog(@"----------------------------------");
        [handler1 handleRequest:700];
        
        NSLog(@"----------------------------------");
        [handler1 handleRequest:10000];

4、输出

2016-12-14 19:48:37.405 责任链模式[65533:2201187] 项目经理同意了费用申请
2016-12-14 19:48:37.406 责任链模式[65533:2201187] ----------------------------------
2016-12-14 19:48:37.406 责任链模式[65533:2201187] 项目经理没有权限批准,转到部门经理处理
2016-12-14 19:48:37.406 责任链模式[65533:2201187] 部门经理同意了费用申请
2016-12-14 19:48:37.406 责任链模式[65533:2201187] ----------------------------------
2016-12-14 19:48:37.406 责任链模式[65533:2201187] 项目经理没有权限批准,转到部门经理处理
2016-12-14 19:48:37.406 责任链模式[65533:2201187] 部门经理没有权限批准,转到总经理处理
2016-12-14 19:48:37.406 责任链模式[65533:2201187] 总经理同意了费用申请
Program ended with exit code: 0

假设这个时候申请流程变了,如下所示:

  • 费用小于1000,由项目经理审批
  • 费用大于1000,由总经理审批

流程里面去掉了部门经理,并且每个经理的审批金额也变了,那么如何更改呢?

很简单,分为两步

  1. 更改原来项目经理和总经理的审批流程
  2. 改变责任链,如下所示
  handler1.successor = handler2;
  handler2.successor = handler3;
改为:
  handler1.successor = handler3;

客户端计算奖金的方式依然不变,这就是责任链的强大之处,可以动态选择构成责任链的成员,看到这里是不是觉得和装饰者模式很类似?装饰器模式也可以动态的选择给原有对象添加各种功能,他们在某种程度上是可以互相替换的,两者都可以实现动态给对象添加功能。

标准的职责链模式是链中有一个对象能处理请求,就停止把请求往下传递,如果不停止传递,那么此时的职责链模式就和装饰器模式很类似了,每个处理对象都相当于一个装饰器。但是他们的目的值不同的,装饰器是为了透明的给对象添加功能,而职责链是为了解耦请求的发送者和接受者


优缺点

  1. 降低耦合度

    该模式使得一个对象无需知道是其他哪一个对象处理其请求。对象仅需
    知道该请求会被“正确”地处理。接收者和发送者都没有对方的明确的信息,且链中的对象不需知道链的结构。
    结果是,职责链可简化对象的相互连接。它们仅需保持一个指向其后继者的引用,而不需保持它所有的候选接受者的引用。

  1. 动态组合职责

    当 在 对 象 中 分 派 职 责 时 , 职 责 链 给 你 更多的灵活性。你可以通过在运行时刻对该链进行动态的增加或修改,来增加或改变处理一个请求的职责链对象。

  1. 不保证被接受

    既然一个请求没有明确的接收者,那么就不能保证它一定会被处理。 该请求可能一直到链的末端都得不到处理。一个请求也可能因该链没有被正确配置而得不到处理。


处理多个请求的职责链

上面的职责链中的每个对象(各个经理)都只处理一种请求,实际情况每个对象可能需要处理多种业务类型。比如现在又增加了差旅费用申请。

常规做法是在抽象类handler中增加一个新的方法,然后在每个具体职责链对象中去实现这个方法,但是这样做就违反了开闭原则,导致每次增加或者修改业务,都需要修改职责链中的每个对象。有没有一种通用的处理办法来处理所有的业务呢?肯定有,具体见demo,这里就不展开说了。


功能链

其实在实际开发中一般都会对职责链进行变通使用,标准的职责链实现是:如果链中对象可以处理请求,就停止把请求传递到下一个对象。变通的处理就是不停止传递,每个对象都参与处理请求,这样的例子很多。

比如各种web过滤器,一个连接请求进来,会进行权限检查,字符集转换等过滤,可以被每个过滤功能都实现为一个职责链对象,然后对于不同的请求就可以动态组合这些职责链对象来构成一条职责链进行过滤。

再比如用户的注册流程一般如下图所示:

image

蓝色方块代表的是一个个的流程,每个箭头流向都是一条链,我们可以把每个流程时限为职责链对象,然后根据不同的流程组合出来不同的职责链来处理注册。

这里只是做抛砖引玉之用,具体代码就不演示了。职责链在实际开发过程中用途广泛,只要是流程类的过程都可以尝试使用职责链去实现。


Demo下载

职责链处理单个请求Demo

职责链处理多个请求Demo

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

推荐阅读更多精彩内容