《Java设计模式》之责任链模式
责任链模式是一种对象的行为模式。在责任链模式里,非常多对象由每个对象对其下家的引用而连接起来形成一条链。请求在这个链上传递,直到链上的某一个对象决定处理此请求。发出这个请求的client并不知道链上的哪一个对象终于处理这个请求,这使得系统能够在不影响client的情况下动态地又一次组织和分配责任。
从击鼓传花谈起
击鼓传花是一种热闹而又紧张的饮酒游戏。在酒宴上宾客依次坐定位置,由一人击鼓,击鼓的地方与传花的地方是分开的。以示公正。開始击鼓时,花束就開始依次传递,鼓声一落,假设花束在某人手中。则该人就得饮酒。
比方说,贾母、贾赦、贾政、贾宝玉和贾环是五个參加击鼓传花游戏的传花者。他们组成一个环链。
击鼓者将花传给贾母,開始传花游戏。花由贾母传给贾赦,由贾赦传给贾政。由贾政传给贾宝玉,又贾宝玉传给贾环。由贾环传回给贾母,如此往复,例如以下图所看到的。当鼓声停止时,手中有花的人就得运行酒令。
击鼓传花便是责任链模式的应用。责任链可能是一条直线、一个环链或者一个树结构的一部分。
责任链模式的结构
以下使用了一个责任链模式的最简单的实现。
责任链模式涉及到的角色例如以下所看到的:
● 抽象处理者(Handler)角色:定义出一个处理请求的接口。
假设须要,接口能够定义 出一个方法以设定和返回对下家的引用。这个角色通常由一个Java抽象类或者Java接口实现。上图中Handler类的聚合关系给出了详细子类对下家的引用。抽象方法handleRequest()规范了子类处理请求的操作。
● 详细处理者(ConcreteHandler)角色:详细处理者接到请求后,能够选择将请求处理掉,或者将请求传给下家。
因为详细处理者持有对下家的引用。因此。假设须要。详细处理者能够訪问下家。
源码
抽象处理者角色
详细处理者角色
client类
能够看出,client创建了两个处理者对象,并指定第一个处理者对象的下家是第二个处理者对象,而第二个处理者对象没有下家。然后client将请求传递给第一个处理者对象。
因为本演示样例的传递逻辑很easy:仅仅要有下家。就传给下家处理。假设没有下家。就自行处理。因此,第一个处理者对象接到请求后,会将请求传递给第二个处理者对象。
因为第二个处理者对象没有下家,于是自行处理请求。活动时序图例如以下所看到的。
使用场景
来考虑这样一个功能:申请聚餐费用的管理。
非常多公司都是这种福利。就是项目组或者是部门能够向公司申请一些聚餐费用,用于组织项目组成员或者是部门成员进行聚餐活动。
申请聚餐费用的大致流程通常是:由申请人先填写申请单。然后交给领导审批,假设申请批准下来,领导会通知申请人审批通过,然后申请人去財务领取费用。假设没有批准下来。领导会通知申请人审批未通过。此事也就此作罢。
不同级别的领导,对于审批的额度是不一样的,比方,项目经理仅仅能审批500元以内的申请;部门经理能审批1000元以内的申请;而总经理能够审核随意额度的申请。
也就是说。当某人提出聚餐费用申请的请求后,该请求会经由项目经理、部门经理、总经理之中的某一位领导来进行对应的处理,可是提出申请的人并不知道终于会由谁来处理他的请求。一般申请人是把自己的申请提交给项目经理。也许最后是由总经理来处理他的请求。
能够使用责任链模式来实现上述功能:当某人提出聚餐费用申请的请求后。该请求会在 项目经理—〉部门经理—〉总经理 这样一条领导处理链上进行传递,发出请求的人并不知道谁会来处理他的请求,每一个领导会依据自己的职责范围,来推断是处理请求还是把请求交给更高级别的领导。仅仅要有领导处理了,传递就结束了。
须要把每位领导的处理独立出来。实现成单独的职责处理对象。然后为它们提供一个公共的、抽象的父职责对象,这样就能够在client来动态地组合职责链。实现不同的功能要求了。
源码
抽象处理者角色类
详细处理者角色
client类
执行结果例如以下所看到的:
责任链模式是对象的行为模式。
使多个对象都有机会处理请求,从而避免请求的发送者和接受者直接的耦合关系。
将这些对象连成一条链。沿着这条链传递该请求,直到有一个对象处理它为止。责任链模式强调的是每个对象及其对下家的引用来组成一条链。利用这样的方式将发送者和接收者解耦,类图例如以下:
也就是说它处理自己能处理的请求且能够訪问它的下家。
package chainOfResp; /** * *作者:alaric *时间:2013-8-17上午11:01:58 *描写叙述:抽象处理角色 */ public abstract class Handler { protected Handler successor; /** * *作者:alaric *时间:2013-8-17上午11:04:22 *描写叙述:处理方法 */ public abstract void handlerRequest(String condition); public Handler getSuccessor() { return successor; } public void setSuccessor(Handler successor) { this.successor = successor; } }
package chainOfResp; /** * *作者:alaric *时间:2013-8-17上午11:25:54 *描写叙述:详细处理角色 */ public class ConcreteHandler1 extends Handler { @Override public void handlerRequest(String condition) { // 假设是自己的责任。就自己处理。负责传给下家处理 if(condition.equals("ConcreteHandler1")){ System.out.println( "ConcreteHandler1 handled "); return ; }else{ System.out.println( "ConcreteHandler1 passed "); getSuccessor().handlerRequest(condition); } } }
package chainOfResp; /** * *作者:alaric *时间:2013-8-17上午11:25:54 *描写叙述:详细处理角色 */ public class ConcreteHandler2 extends Handler { @Override public void handlerRequest(String condition) { // 假设是自己的责任,就自己处理,负责传给下家处理 if(condition.equals("ConcreteHandler2")){ System.out.println( "ConcreteHandler2 handled "); return ; }else{ System.out.println( "ConcreteHandler2 passed "); getSuccessor().handlerRequest(condition); } } }
package chainOfResp; /** * *作者:alaric *时间:2013-8-17上午11:25:54 *描写叙述:详细处理角色 */ public class ConcreteHandlerN extends Handler { /** * 这里如果n是链的最后一个节点必须处理掉 * 在实际情况下。可能出现环,或者是树形。 * 这里并不一定是最后一个节点。 * */ @Override public void handlerRequest(String condition) { System.out.println( "ConcreteHandlerN handled"); } }
package chainOfResp; /** * *作者:alaric *时间:2013-8-17上午10:59:06 *描写叙述:測试类 */ public class Client { /** *作者:alaric *时间:2013-8-17上午10:58:58 *描写叙述: */ public static void main(String[] args) { Handler handler1 = new ConcreteHandler1(); Handler handler2 = new ConcreteHandler2(); Handler handlern = new ConcreteHandlerN(); //链起来 handler1.setSuccessor(handler2); handler2.setSuccessor(handlern); //如果这个请求是ConcreteHandler2的责任 handler1.handlerRequest("ConcreteHandler2"); } }
能够看出它们每一个节点都能够做一些事情。所以不算一个纯的责任链。
package chainOfResp.example; /** * *作者:alaric *时间:2013-8-17下午1:02:51 *描写叙述:审批处理抽象类 */ public abstract class Handler { protected Handler handler; /** * *作者:alaric *时间:2013-8-17下午1:07:40 *描写叙述:审批 */ public abstract boolean approve(double day); public Handler getHandler() { return handler; } public void setHandler(Handler handler) { this.handler = handler; } }
package chainOfResp.example; public class GroupLeader extends Handler { @Override public boolean approve(double day) { if(day<0.5){ System.out.println("小组长审批通过"); return true; }else { System.out.println("小组长传给了他的上司"); return getHandler().approve(day); } } }
package chainOfResp.example; public class ProjectManager extends Handler { @Override public boolean approve(double day) { if(day<2){ System.out.println("项目经理审批通过"); return true; }else { System.out.println("项目经理传给了他的上司"); return getHandler().approve(day); } } }
package chainOfResp.example; public class DepartmentManager extends Handler { @Override public boolean approve(double day) { if(day<5){ System.out.println("部门经理审批通过"); return true; }else { System.out.println("部门经理传给了他的上司"); return getHandler().approve(day); } } }
package chainOfResp.example; public class CEO extends Handler { @Override public boolean approve(double day) { System.out.println("部门经理审批通过"); return true; } }
package chainOfResp.example; /** * *作者:alaric *时间:2013-8-17下午12:54:51 *描写叙述:測试类。首先来创建责任链,然后发出请求模拟员工来请假 */ public class Client { /** *作者:alaric *时间:2013-8-17下午12:54:44 *描写叙述: */ public static void main(String[] args) { //创建节点 GroupLeader gl = new GroupLeader(); ProjectManager pm = new ProjectManager(); DepartmentManager dm = new DepartmentManager(); CEO ceo = new CEO(); //建立责任链 gl.setHandler(pm); pm.setHandler(dm); dm.setHandler(ceo); //向小组长发出申请,请求审批4天的假期 gl.approve(4D); } }执行结果:
本文借鉴:
http://www.cnblogs.com/java-my-life/archive/2012/05/28/2516865.html
http://www.tuicool.com/articles/RJvARj