参考文章:http://wiki.jikexueyuan.com/project/design-pattern-behavior/chain-four.html
定义:
使多个对象都有机会处理请求,从而避免请求的发送者和接收者之间的耦合关系。将这些对象连成一条链,并沿着这条链传递该请求,直到有一个对象处理它为止。
分类:
-
纯的职责链模式
一个纯的职责链模式要求一个具体处理者对象只能在两个行为中选择一个:要么承担全部责任,要么将责任推给下家,不允许出现某一个具体处理者对象在承担了一部分或全部责任后又将责任向下传递的情况。而且在纯的职责链模式中,要求一个请求必须被某一个处理者对象所接收,不能出现某个请求未被任何一个处理者对象处理的情况。在前面的采购单审批实例中应用的是纯的职责链模式。
-
不纯的职责链模式
在一个不纯的职责链模式中允许某个请求被一个具体处理者部分处理后再向下传递,或者一个具体处理者处理完某请求后其后继处理者可以继续处理该请求,而且一个请求可以最终不被任何处理者对象所接收。
uml类图:
模式组成:
Handler
: 抽象处理者角色,声明一个请求处理的方法,并保持下一个处理节点Handler
对象的引用.ConcreteHandler
: 具体处理者角色,对请求进行处理,如果不能处理,则将请求传递给下一个节点Handler
优点:
- 责任链模式减低了发出命令的对象和处理命令的对象之间的耦合
- 发出命令的对象只是把命令传给链结构的起始者,而不需要知道到底是链上的哪一个节点处理了这个命令。
- 在给对象分派职责时,职责链可以给我们更多的灵活性,可以通过在运行时对该链进行动态的增加或修改来增加或改变处理一个请求的职责。
- 在系统中增加一个新的具体请求处理者时无须修改原有系统的代码,只需要在客户端重新建链即可,从这一点来看是符合“开闭原则”的。
缺点:
- 由于一个请求没有明确的接收者,那么就不能保证它一定会被处理,该请求可能一直到链的末端都得不到处理;一个请求也可能因职责链没有被正确配置而得不到处理。
- 所有责任开始都是从指定的接收者依次传递下去,运行中无法从中间开始传递
应用场景:
- 有多个对象可以处理同一个请求,具体哪个对象处理该请求待运行时刻再确定,客户端只需将请求提交到链上,而无须关心请求的处理对象是谁以及它是如何处理的。
- 需要动态指定一组对象处理请求
实际应用:
- Java的异常机制就是一个责任链模式,一个try可以对应多个cathc。如果某一个catch不匹配,则跳到下一个catch中
- Servlet开发中,过滤器的链式处理
- Struts2中,拦截器的处理
举个栗子:
(审批流程)
定义抽象处理角色
interface Handler { public boolean handleRequest(Request request); public String getName(); }
定义一个请求类
class Request { public int getLeaveDays() { return 4; } }
定义一个具体处理角色
class Manager implements Handler { @Override public boolean handleRequest(Request request) { if (request.getLeaveDays() > 5) { return false; } return true; } @Override public String getName() { return "部门经理"; } }
定义一个具体处理角色
class CTO implements Handler { @Override public boolean handleRequest(Request request) { if (request.getLeaveDays() > 3) { return false; } return true; } @Override public String getName() { return "技术总监"; } }
初始化责任链,并调用
public class ChainOfResponsibilityTest { static List<Handler> handlers = new ArrayList<Handler>(); static{ handlers.add(new Manager()); handlers.add(new CTO()); } public static void main(String[] args) { for (Handler h : handlers) { if (!h.handleRequest(new Request())) { System.out.println("该次请假未通过审批,审批人:"+h.getName()); } } } }
输出
时间: 2024-10-13 16:19:10