责任链模式

注意:参考学java的生生责任链模式,代码

一.基本框架

如上图所示,责任链的由节点(实现了ValidatorHandler接口)和主流程(负责组装各个节点)组成.

其实现方式一共有两种:

  1. 数组:如上图所示,在Validator中通过List将各个节点存储起来.
  2. 链表: 在各个节点中存储next,并提供appNext(ValidatorHandler handler)方法供Validator装配使用.

二. 引入上下文

在责任连中引入上下文,对外不可见,只对责任链的主流程和各个节点可见;可以方便存储每个节点过滤的信息,同时,也便于对主流程序进行干预

1. 升级1:责任链的所有节点全部执行

需求:在责任链的基本实现中,当前面节点抛出异常,后面节点便不会再执行,如果需要责任链全部执行完然后再抛出全部的异常,如何进行?

方案1:不引入上下文,在主流程中直接捕获异常存储起来,当节点全部执行完毕后,再抛出异常.

java 复制代码
public class ValidatorChain {

    private final List<ValidatorHandler> handlers = new ArrayList<>();

    public void validate(Object value) {
    	//存储异常信息
        List<String> list = new ArrayList<>();
        for (ValidatorHandler handler : handlers) {
            try {
                handler.validate(value, context);
            }catch (ValidatorException e){
                list.add(e.getMessage());
            }
        }
        if (list.isEmpty()) {
            return;
        }
        throw new ValidatorException(list.toString());
    }
   
}

方案2:引入上下文,节点不再抛出异常,直接将异常信息存储在上下文中,执行完全部节点后再抛出异常

java 复制代码
public class ValidatorContext {
    private final List<String> errorMessageList = new ArrayList<>();
    public void appendError(String message) {
        errorMessageList.add(message);
    }
    public void throwExceptionIfNecessary() {
        if(errorMessageList.isEmpty()){
            return;
        }
        throw new ValidatorException(errorMessageList.toString());
    }
}
java 复制代码
public class ValidatorChain {

    private final List<ValidatorHandler> handlers = new ArrayList<>();

    public void validate(Object value) {
        ValidatorContext context = new ValidatorContext();
        for (ValidatorHandler handler : handlers) {
            handler.validate(value, context);
            if(context.isStopped()){
                break;
            }
        }
        context.throwExceptionIfNecessary();
    }

}

2. 升级2:节点干预主流程

需求: 在升级1的基础上,如何实现节点控制当主流程的进行?

解决方案:在上下文中引入stopped变量,如果某一个节点想要中断流程,则调用上下文的stopChain()方法,同时在主流程中执行完该节点后取出上下文的stopChain方法判断主流成是否中断

上下文

java 复制代码
public class ValidatorContext {
    private final List<String> errorMessageList = new ArrayList<>();
    private boolean stopped =false;

    private int currentIndex = 0;

    public void stopChain(){
        this.stopped =true;
    }
    public boolean isStopped(){
        return stopped;
    }
    
    public void throwExceptionIfNecessary() {
        if(errorMessageList.isEmpty()){
            return;
        }
        throw new ValidatorException(errorMessageList.toString());
    }

}

主流程:

java 复制代码
public class ValidatorChain {

    private final List<ValidatorHandler> handlers = new ArrayList<>();

    public void validate(Object value) {
        ValidatorContext context = new ValidatorContext();
        for (ValidatorHandler handler : handlers) {
            handler.validate(value, context);
            if(context.isStopped()){
                break;
            }
        }
        context.throwExceptionIfNecessary()
    }
    public void addLastHandler(ValidatorHandler handler) {
        this.handlers.add(handler);

    }
}

节点:

java 复制代码
public class MaxValidatorHandler  implements ValidatorHandler{

    private final int max;

    public MaxValidatorHandler(int max) {
        this.max = max;
    }

    @Override
    public void validate(Object value,ValidatorContext context) {
        if(value instanceof Integer integerValue){
            if(integerValue > max){
                context.appendError("你的值是"+ integerValue+ "不能大于"+max);
                context.stopChain();
            }
        }
    }
}
相关推荐
__万波__3 天前
二十三种设计模式(二十三)--责任链模式
java·设计模式·责任链模式
罗小爬EX7 天前
杂记 - 状态模式 VS. 责任链模式
状态模式·责任链模式
山风wind11 天前
Spring中责任链模式的工业级应用简单剖析
java·spring·责任链模式
山风wind11 天前
设计模式-责任链模式:让请求在链条中流动直到被处理
设计模式·责任链模式
JavaBoy_XJ13 天前
行为型-责任链模式
责任链模式
山沐与山16 天前
【设计模式】Python责任链模式:从入门到实战
python·设计模式·责任链模式
老朱佩琪!16 天前
Unity责任链模式
unity·设计模式·责任链模式
吃不饱的得可可16 天前
【Linux】System V消息队列与责任链模式
linux·运维·责任链模式
o0向阳而生0o18 天前
116、23种设计模式之责任链模式(23/23)(完结撒花)
设计模式·责任链模式
吃喝不愁霸王餐APP开发者19 天前
利用责任链模式解耦多平台(美团/饿了么)霸王餐接口的适配逻辑
android·责任链模式