设计模式-责任链模式

1. 责任链- 使用校验的责任链 完成校验逻辑

责任链模式 其核心在将逻辑抽象成一条链条 (可维护的)核心思想是 每个节点都能干预主流程的执行 建立统一的上下文管理不同字段的校验链 主流程汇聚每个节点的结果。

1.1 首先定义好注解,并在实体类中添加注解
java 复制代码
@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
public @interface Length {

    int value();
}
java 复制代码
@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
public @interface Max {

    int value();
}
java 复制代码
@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
public @interface Min {

    int value();
}
java 复制代码
public class User {
    public String getName() {
        return name;
    }

    public User(String name, Integer age) {
        this.name = name;
        this.age = age;
    }

    public void setName(String name) {
        this.name = name;
    }

    public Integer getAge() {
        return age;
    }

    public void setAge(Integer age) {
        this.age = age;
    }

    @Length(5)
    private String name;

    @Max(15)
    @Min(17)
    private Integer age;
}

设计出整个校验链,使用 hanlders 做为校验处理器的List

java 复制代码
    private final List<ValidatorHandler> handlers = new ArrayList<>();

封装出 addHandler的方法 用来向外暴露添加校验处理器,并以顺序作为优先级

java 复制代码
   public void addLastHandler(ValidatorHandler handler) {
        this.handlers.add(handler);
    }
1.2 首先抽象出校验的类和方法

在 validate 方法中添加校验的对象,通过反射的方式对字段中需要做的校验抽象出来 ,并构建出校验的链条

java 复制代码
public class Validator {

   public  void validate(Object bean) throws ValidatorException, IllegalAccessException {
         // 责任链模式
        Class<?> aClass = bean.getClass();

        for (Field declaredField : aClass.getDeclaredFields()) {
            // 私有属性添加
            declaredField.setAccessible(true);
            buildValidateChain(declaredField).validate(declaredField.get(bean));
        }

    }

    private ValidatorChain buildValidateChain(Field declaredField) {
        ValidatorChain chain = new ValidatorChain();

        Max max = declaredField.getAnnotation(Max.class);
        if ( max != null) {
            chain.addLastHandler(new MaxValidatorHandler(max.value()));
        }
        Min min = declaredField.getAnnotation(Min.class);
        if (min != null) {
            chain.addLastHandler( new MinValidatorHandler(min.value()));
        }
        Length length = declaredField.getAnnotation(Length.class);
        if (length != null) {
            chain.addLastHandler(new LengthValidatorHandler(length.value()));
        }

        return chain;
    }
}
1.3 ValidatorHandler 作为校验处理器 对各种校验逻辑的封装
java 复制代码
public interface ValidatorHandler {

    void validate(Object bean,ValidatorContext context) ;
}


//  各种不同的校验器实现 
public class MaxValidatorHandler implements ValidatorHandler {
    //
    private final Integer max;

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


    // 校验器的告警机制停止校验 每个校验器都可以通过上下文对流程进行干预
    @Override
    public void validate(Object bean, ValidatorContext context) {
        
        if(bean instanceof Integer) {
            if(max < (Integer) bean) {
                context.appendError("Max number is greater than " + max); 
        } 

    }

}

// 最小值的校验
public class MinValidatorHandler implements ValidatorHandler {

    private final Integer min;

    public MinValidatorHandler (Integer min ) {
       this.min = min;
    }

    @Override
    public void validate(Object bean,  ValidatorContext context) {

        if(bean instanceof Integer) {
            if(min > (Integer) bean) {
                context.appendError("Max number is min than " + min);
            }
        }

    }
}


// 长度的校验
public class LengthValidatorHandler implements ValidatorHandler {

    private final Integer length;

    public LengthValidatorHandler(Integer length ) {
       this.length = length;
    }

    @Override
    public void validate(Object bean,ValidatorContext context) throws ValidatorException {

        if(bean instanceof String) {
            if(length < ((String) bean).length()) {
                    context.appendError("Max length is greater than " + length);
            }
        }

    }
}
1.4 在具体的校验方法中的处理

在 validatorContext对象中封装一些公用的方法(如当前校验到的节点的下表,当前错误的信息的List的封装)和当前校验节点的下标。使用 ValidatorHandler的实现各种校验逻辑的封装,进行校验。

java 复制代码
    // 组装链条
    public void validate(Object object) throws ValidatorException {

//      for (ValidatorHandler handler : handlers) {
//               handler.validate(object,context);
//               if(context.shouldStopChain()){
//                   break;
//               }
//        }
         ValidatorContext context = new ValidatorContext(object);
        while(true) {
            Integer currentIndex = context.getCurrentIndex();
            if(currentIndex == handlers.size() ) {
                break;
            }
            ValidatorHandler validatorHandler = handlers.get(currentIndex);
            validatorHandler.validate(context.getValue(),context);
            // 没有调用 doNext函数的情况
            if(currentIndex.equals(context.getCurrentIndex())){
               break;
            }
        }

        context.throwExceptionIfNecessary();
    }
相关推荐
草明2 分钟前
docker stats 增加一列容器名称的显示
java·开发语言·docker
期待のcode6 分钟前
Maven的概念与Maven项目的创建
java·maven
我崽不熬夜20 分钟前
Java中的基本数据类型和包装类:你了解它们的区别吗?
java·后端·java ee
我是廖志伟1 小时前
【jar包启动,每天生成一个日志文件】
java·jar
牛奶咖啡131 小时前
学习设计模式《二十三》——桥接模式
学习·设计模式·桥接模式·认识桥接模式·桥接模式的优点·何时选用桥接模式·桥接模式的使用示例
掉鱼的猫1 小时前
Solon StateMachine 实现状态机使用示例详解
java·状态机
用户6135411460161 小时前
BurpSuite 1.4.07.jar 怎么使用?详细安装和抓包教程(附安装包下载)
java
ankleless1 小时前
Spring 框架深度解析:从核心原理到实战应用
java·spring
带刺的坐椅1 小时前
Spring AOP 与 Solon AOP 有什么区别?
java·spring·solon·aop
左灯右行的爱情1 小时前
深度学习设计模式:责任链(Chain of Responsibility)模式(例子+业务场景+八股)
深度学习·设计模式·责任链模式