责任链模式

一、定义

定义 : 为了避免请求的发送者和接收者之间的耦合关系,使多个接受对象都有机会处理请求。将这些对象连成一条链,并沿着这条链传递该请求,直到有一个对象处理它为止。责任链模式是一种对象行为型模式。

由定义上可以看出,责任链模式就是将一堆可以处理请求的对象连成一条链,然后一个一个试着处理请求。

二、责任链模式的结构

image

在责任链模式结构图中包含如下几个角色:

Handler(抽象处理者):定义一个处理请求的接口,一般为抽象类,请求处理方法为抽象方法,不同的具体处理者实现具体的请求处理。因为每一个处理者的下家还是一个处理者,因此在抽象处理者中定义了一个抽象处理者类型的对象,作为其对下家的引用。通过该引用,处理者可以连成一条链。

ConcreteHandler(具体处理者):它是抽象处理者的子类,可以处理用户请求,在具体处理者类中实现了抽象处理者中定义的抽象请求处理方法,在处理请求之前需要进行判断,看是否有相应的处理权限,如果可以处理请求就处理它,否则将请求转发给后继者;在具体处理者中可以访问链中下一个对象,以便请求的转发。

三、经典模式的实现

1.定义AbstractHandler(抽象处理者),使子类形成一条链

public abstract class AbstractHandler {
    private AbstractHandler handler;

    public abstract void handleRequest(String condition);

    public AbstractHandler getHandler() {
        return handler;
    }

    public void setHandler(AbstractHandler handler) {
        this.handler = handler;
    }
}

2.创建若干个ConcreteHandler(具体处理者)继承AbstractHandler,在当前处理者对象无法处理时,将执行权传给下一个处理者对象

public class ConcreteHandlerA extends AbstractHandler {

    @Override
    public void handleRequest(String condition) {
        if (condition.equals("A")) {
            System.out.println("ConcreteHandlerA处理");
        } else {
            System.out.println("ConcreteHandlerA不处理,由其他的Handler处理");
            super.getHandler().handleRequest(condition);
        }
    }
}

public class ConcreteHandlerB extends AbstractHandler {

    @Override
    public void handleRequest(String condition) {
        if (condition.equals("B")) {
            System.out.println("ConcreteHandlerB处理");
        } else {
            System.out.println("ConcreteHandlerB不处理,由其他的Handler处理");
            super.getHandler().handleRequest(condition);
        }
    }
}

public class ConcreteHandlerZ extends AbstractHandler {
    @Override
    public void handleRequest(String condition) {
        //一般是最后一个处理者
        System.out.println("ConcreteHandlerZ处理");
    }
}

3.创建ChainClient(测试类)

public class ChainClient {

    public static void main(String[] args) {
        AbstractHandler handlerA = new ConcreteHandlerA();
        AbstractHandler handlerB = new ConcreteHandlerB();
        AbstractHandler handlerZ = new ConcreteHandlerZ();
        // 如A处理不掉转交给B
        handlerA.setHandler(handlerB);
        handlerB.setHandler(handlerZ);
        handlerA.handleRequest("Z");
    }
}

4.运行效果

----------------------handleRequest("A")-------------------------
ConcreteHandlerA处理
----------------------handleRequest("B")-------------------------
ConcreteHandlerA不处理,由其他的Handler处理
ConcreteHandlerB处理
----------------------handleRequest("Z")-------------------------
ConcreteHandlerA不处理,由其他的Handler处理
ConcreteHandlerB不处理,由其他的Handler处理
ConcreteHandlerZ处理

纯与不纯

  • 纯:一个纯的责任链模式要求:

1.一个具体处理者对象只能在“处理请求”和“转发请求”中两选一。

  1. 一个请求必须被某一个处理者对象所接收,不能出现某个请求未被任何一个处理者对象处理的情况。
  • 不纯:在一个不纯的责任链模式中则允许:
  1. 一个请求应该被所有的对象处理,直到某一个处理失败。(例如下面分析校验模块)
  2. 或者一个具体处理者处理完某请求后其后继处理者可以继续处理该请求。(dubbo filter)
  3. 或者一个请求可以最终不被任何处理者对象所接收。

纯的责任链模式的实际例子很难找到,一般看到的例子均是不纯的责任链模式的实现。

不纯的实现案列

背景:最近在梳理一块业务,目前面临的一个问题是数据入库前需要做很多校验操作,例如:

  1. 查询数据库判断数据是不是已经存在;
  2. rpc调用别的服务判断某个字段是都有效
  3. 等等

如果所有校验代码写在一个方法中感觉太臃肿,可读性太差,不易扩展。现在准备使用不纯的责任链模式重构一版本。根据不同的校验条件抽出来不同的校验类。
既然是校验操作,所有的校验组成一条链,请求挨个校验,直到有一个校验失败,或者全部校验成功才会终止。

1.定义Verify(抽象处理者,使子类形成一条链

public interface Verify {

    Message doVerity(AddForm form);

}

2.创建若干个Verify实现者(具体处理者)继承AbstractHandler,在当前处理者处理成功后,继续往下处理

@Service
@Order(1)
public class VerifyA implements Verify {
    @Override
    Message doVerity(AddForm form) {
        //执行自己的校验逻辑
        boolean isPass = xxxxxA();
        if (isPass) {
            return Message.ok();
        } else {
            return Message.error(RstStatus.invalidParameter, "校验不通过的原因");
        }
    }
}

@Service
@Order(2)
public class VerifyB implements Verify {
    @Override
    Message doVerity(AddForm form) {
        //执行自己的校验逻辑
        boolean isPass = xxxxxB();
        if (isPass) {
            return Message.ok();
        } else {
            return Message.error(RstStatus.invalidParameter, "校验不通过的原因");
        }
    }
}

             VerifyC
             VerifyD
              ...

3.spring 环境下调用责任链

 /**
     * IOC 自动组成所有实现类的调用链
     * 实现类可通过@Order进行排序
     * **/
    @Resource
    private List<Verify> verifyList;

    public Message isParamValid(AddForm addForm) {
        if (CollectionUtils.isEmpty(verifyList)) {
            return Message.ok();
        }
        for (Verify verify : verifyList) {
            Message message = verify.doVerity(addForm);
            //校验通过继续循环
            if (RstStatus.ok.getCode() == message.getCode()) {
                continue;
            }
            //失败直接返回
            return message;
        }
    }

好处

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

推荐阅读更多精彩内容