设计模式--责任链(职责链)模式

学校OA系统的采购审批项目,需求是:

1) 采购员采购教学器材。

2) 如果金额小于等于5000,由教学主任审批。

3) 如果金额小于等于10000,由院长审批。

4) 如果金额小于等于30000,由副校长审批。

5) 如果金额超过30000,由校长审批。

请设计程序完成采购审批项目。

传统方案(类图)如下:

传统方案分析以及带来的问题:

1) 传统方案是:接收到一个采购请求后,根据采购金额来调用对应的Approve审批人来完成审批。

2) 传统方案分析:客户端会使用到分支判断(比如switch)来对不同的采购请求处理。这样就存在问题:

一、 如果每个审批人的审批金额发生了变化,客户端也需要变化。

二、 客户端必须明确的知道有多少个审批级别和访问。

3) 这样,对一个采购请求处理和Approver审批人会存在强烈的耦合关系,不利于代码的维护和扩展。

4) 解决方案:责任链模式。

责任链模式(Chain of Responsibility Pattern)介绍:

1. 责任链模式(Chain of Responsibility Pattern)又叫责任链,为请求创建一个接收者对象的链条(示意图),这种模式对请求的发送者和接受者解耦。

2. 责任链模式通常每个接收者都包含了对另一个接收者的引用。如果该对象不能接收请求,则会把相同的请求传给下一个接收者,以此类推。

3. 责任链模式也是属于行为模式。

责任链模式类图:

对原理类图的说明即责任链模式的角色和职责:

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

2) Handler:抽象的处理者,定义了一个处理请求的接口和方法,它是一个接口或者一个抽象类。

3) AbstractHandler:抽象类实现了Handler,同时含有另外一个handler对象。

4) ConcreteHandler:是具体的处理者,它处理它负责的请求,他还可以访问它的后继者(即下一个处理者)。如果可以处理当前请求则自己处理,如果不能则将该请求交给下一个处理者处理,从而形成一个职责链。

5) Request:含有多个属性,表示一个请求。

责任链模式解决学校采购问题:

1. URL类图:

2. 实现代码:

package com.xia.designmode.study.chainofresponsibilitypattern;

/**

* 责任链模式--处理接口(自己加的)

* */

public interface Approver {

/*

  * 处理请求方法,得到一个请求,处理是由具体实现类来完成。

  * */

    public void processRequest(PurchaseRequest purchaseRequest);

}

package com.xia.designmode.study.chainofresponsibilitypattern;

/**

* 责任链模式--抽象处理类角色

* */

public abstract class AbstractApprover implements Approver {

    private Approver nextApprover;

    private String name;

    public AbstractApprover(String name) {

        this.name=name;

    }

    /*

    *下一个处理者

    * */

    public void setNextApprover(Approver nextApprover) {

        this.nextApprover = nextApprover;

    }

    public Approver getNextApprover() {

        return nextApprover;

    }

    public String getName() {

        return name;

    }

}

package com.xia.designmode.study.chainofresponsibilitypattern;

import java.math.BigDecimal;

/**

* 责任链模式--具体处理类角色

* */

public class CollegeApprover extends AbstractApprover{

    public CollegeApprover(String name) {

        super(name);

    }

    @Override

    public void processRequest(PurchaseRequest purchaseRequest) {

        //处理请求,如果金额小于等于10000,则处理

        BigDecimal price=purchaseRequest.getPrice();

        if(price.compareTo(new BigDecimal(10000))<=0) {

            System.out.println("请求编号number:"+purchaseRequest.getNumber()+"由"+this.getName() + "处理");

        }else{

            this.getNextApprover().processRequest(purchaseRequest);

        }

    }

}

package com.xia.designmode.study.chainofresponsibilitypattern;

import java.math.BigDecimal;

/**

* 责任链模式--具体处理类角色

* */

public class DepartmentApprover extends AbstractApprover{

    public DepartmentApprover(String name) {

        super(name);

    }

    @Override

    public void processRequest(PurchaseRequest purchaseRequest) {

        //处理请求,如果金额小于等于5000,则处理

        BigDecimal price=purchaseRequest.getPrice();

        if(price.compareTo(new BigDecimal(5000))<=0) {

            System.out.println("请求编号id:"+purchaseRequest.getId()+"由"+this.getName() + "处理");

        }else{

            this.getNextApprover().processRequest(purchaseRequest);

        }

    }

}

package com.xia.designmode.study.chainofresponsibilitypattern;

import java.math.BigDecimal;

/**

* 责任链模式--具体处理类角色

* */

public class HeadmasterApprover extends AbstractApprover{

    public HeadmasterApprover(String name) {

        super(name);

    }

    @Override

    public void processRequest(PurchaseRequest purchaseRequest) {

        //其他人处理不了的,校长来处理。

        System.out.println("请求编号id:"+purchaseRequest.getId()+"由"+this.getName() + "处理");

    }

}

package com.xia.designmode.study.chainofresponsibilitypattern;

import java.math.BigDecimal;

/**

*责任链模式-Request请求角色

*

* */

public class PurchaseRequest {

    private String type; //请求类型

    private String number;//请求单号

    private BigDecimal price;//价钱

    private int id;//id号

    public PurchaseRequest(String type, String number, BigDecimal price, int id) {

        this.type = type;

        this.number = number;

        this.price = price;

        this.id = id;

    }

    public PurchaseRequest(String type, String number, BigDecimal price) {

        this.type = type;

        this.number = number;

        this.price = price;

    }

    public String getType() {

        return type;

    }

    public String getNumber() {

        return number;

    }

    public BigDecimal getPrice() {

        return price;

    }

    public int getId() {

        return id;

    }

}

package com.xia.designmode.study.chainofresponsibilitypattern;

import java.math.BigDecimal;

/**

* 责任链模式--具体处理类角色

* */

public class VicePresidentApprover extends AbstractApprover{

    public VicePresidentApprover(String name) {

        super(name);

    }

    @Override

    public void processRequest(PurchaseRequest purchaseRequest) {

        //处理请求,如果金额小于等于10000,则处理

        BigDecimal price=purchaseRequest.getPrice();

        if(price.compareTo(new BigDecimal(30000))<=0) {

            System.out.println("请求编号id:"+purchaseRequest.getId()+"由"+this.getName() + "处理");

        }else{

            this.getNextApprover().processRequest(purchaseRequest);

        }

    }

}

责任链模式在Spring MVC框架中应用的源码分析:

1) SpringMVC中HandlerExceptionChain类就使用到了责任链模式。

2) SpringMVC请求流程简图。

3) 代码分析+Debug源码+说明。

protected void doDispatch(HttpServletRequest request, HttpServletResponse response) throws Exception {

HandlerExecutionChain mappedHandler = null;

mappedHandler = getHandler(processedRequest);//获取HandlerExecutionChain对象。

//在mappedHandler.applyPreHandle方法内部获得HandlerInterceptor interceptor对象。

//调用了拦截器的interceptor.preHandle方法

if (!mappedHandler.applyPreHandle(processedRequest, response)) {

return;

}

mappedHandler.applyPostHandle(processedRequest, response, mv);

//mappedHandler.applyPostHandle方法中再次获得获取到拦截器对象HandlerInterceptor interceptor,并调用拦截器的interceptor.postHandle方法。

for (int i = interceptors.length - 1; i >= 0; i--) {

HandlerInterceptor interceptor = interceptors[i];

interceptor.postHandle(request, response, this.handler, mv);

}

//在mappedHandler.applyPreHandle方法中还调用了triggerAfterCompletion方法,该方法中还调用了拦截器处理: HandlerInterceptor interceptor = interceptors[i];

try {

interceptor.afterCompletion(request, response, this.handler, ex);

}

catch (Throwable ex2) {

logger.error("HandlerInterceptor.afterCompletion threw exception", ex2);

}

void triggerAfterCompletion(HttpServletRequest request, HttpServletResponse response, @Nullable Exception ex)

throws Exception {

HandlerInterceptor[] interceptors = getInterceptors();

if (!ObjectUtils.isEmpty(interceptors)) {

for (int i = this.interceptorIndex; i >= 0; i--) {

HandlerInterceptor interceptor = interceptors[i];

try {

interceptor.afterCompletion(request, response, this.handler, ex);

}

catch (Throwable ex2) {

logger.error("HandlerInterceptor.afterCompletion threw exception", ex2);

}

}

}

}

说明:

1. Spring MVC请求的流程中,执行了拦截器的相关方法interceptor.postHandle等等。

2. 在处理SpringMVC请求时,使用到了职责链模式以及前面讲到的适配器模式。

3. HandlerExceptionChain主要负责的是请求拦截器的执行和请求处理,但是它本身不处理请求,只是将请求分配给链上注册的处理器执行,这是责任链的执行方式,减少了责任链和处理逻辑之间的耦合,规范了流程。

4. HandlerExceptionChain维护了HandlerInterceptor[](拦截器数组),可以向其中注册相应的拦截器。

责任链模式注意事项和细节:

1. 将请求和处理分开,实现解耦,提高系统的灵活性。

2. 简化了对象,使得对象不需要知道链的结构。

缺点:

1) 性能受到了影响,特别是在链比较长的时候,因此需要控制链中最大节点数量。通常在Handler中设置一个最大节点数量,在setNext()方法中判断是否已经超过阈值,超过则不允许该链条建立,避免过长的链条无意识的破坏系统性能。

2) 调试不方便,采用了类似递归的方式,调试时逻辑可能比较复杂。

适用场景:

1. 有多个对象可以处理多个请求时,比如多级请求、请假/加薪等审批流程;

2. java web中tomcat的encoding处理、拦截器;

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

推荐阅读更多精彩内容