责任链模式(Chain of Responsibility)是一种处理请求的模式,它让多个处理器都有机会处理该请求,直到其中某个处理成功为止。责任链模式把多个处理器串成链,然后让请求在链上传递。
在实际场景中,财务审批就是一个责任链模式。假设某个员工需要报销一笔费用,审核者可以分为:
- Manager:只能审核1000元以下的报销;
- Director:只能审核10000元以下的报销;
- CEO:可以审核任意额度。
用责任链模式设计此报销流程时,每个审核者只关心自己责任范围内的请求,并且处理它。对于超出自己责任范围的,扔给下一个审核者处理,这样,将来继续添加审核者的时候,不用改动现有逻辑。
我们来看看如何实现责任链模式。
首先,我们要抽象出请求对象,它将在责任链上传递:
public class Request {private String name;private BigDecimal amount;public Request(String name, BigDecimal amount) {this.name = name;this.amount = amount;}public String getName() {return name;}public BigDecimal getAmount() {return amount;}}
其次我们要抽象出处理器
public interface Handler {// 返回Boolean.TRUE = 成功// 返回Boolean.FALSE = 拒绝// 返回null = 交下一个处理Boolean process(Request request);}
并且做好约定:如果返回Boolean.TRUE,表示处理成功,如果返回Boolean.FALSE,表示处理失败(请求被拒绝),如果返回null,则交由下一个Handler处理。
然后,依次编写ManagerHandler、DirectorHandler和CEOHandler。以ManagerHandler为例:
public class ManagerHandler implements Handler {public Boolean process(Request request) {// 如果超过1000元,处理不了,交下一个处理:if (request.getAmount().compareTo(BigDecimal.valueOf(1000)) > 0) {return null;}// 对Bob有偏见:return !request.getName().equalsIgnoreCase("bob");}}
有了不同的Handler后,我们还要把这些Handler组合起来,变成一个链,并通过一个统一入口处理:
public class HandlerChain {// 持有所有Handler:private List<Handler> handlers = new ArrayList<>();public void addHandler(Handler handler) {this.handlers.add(handler);}public boolean process(Request request) {// 依次调用每个Handler:for (Handler handler : handlers) {Boolean r = handler.process(request);if (r != null) {// 如果返回TRUE或FALSE,处理结束:System.out.println(request + " " + (r ? "Approved by " : "Denied by ") + handler.getClass().getSimpleName());return r;}}throw new RuntimeException("Could not handle request: " + request);}}
现在,我们就可以在客户端组装出责任链,然后用责任链来处理请求:
// 构造责任链:HandlerChain chain = new HandlerChain();chain.addHandler(new ManagerHandler());chain.addHandler(new DirectorHandler());chain.addHandler(new CEOHandler());// 处理请求:chain.process(new Request("Bob", new BigDecimal("123.45")));chain.process(new Request("Alice", new BigDecimal("1234.56")));chain.process(new Request("Bill", new BigDecimal("12345.67")));chain.process(new Request("John", new BigDecimal("123456.78")));
责任链模式本身很容易理解,需要注意的是,Handler添加的顺序很重要,如果顺序不对,处理的结果可能就不是符合要求的。
此外,责任链模式有很多变种。有些责任链的实现方式是通过某个Handler手动调用下一个Handler来传递Request,例如:
public class AHandler implements Handler {private Handler next;public void process(Request request) {if (!canProcess(request)) {// 手动交给下一个Handler处理:next.process(request);} else {...}}}
