美文网首页
聊聊如何实现一个特别的责任链

聊聊如何实现一个特别的责任链

作者: linyb极客之路 | 来源:发表于2024-10-14 09:27 被阅读0次

前言

什么是责任链

责任链是一种设计模式,它让多个对象有机会处理同一个请求,这些对象形成一个链。请求从链的一端开始,逐个传递给链上的对象,直到某个对象处理它或者请求未被处理。这样,发送请求者无需知道哪个对象会处理,实现了发送者与接收者的解耦,增加了系统的灵活性

责任链的常用场景

  • 权限与认证系统:在登录认证、权限验证流程中,不同的处理者可以检查用户名密码、验证权限级别、处理单点登录等。请求(如访问资源)沿着责任链传递,直到找到合适的处理器来授权或拒绝访问。
  • 日志记录与错误处理:根据日志级别(如DEBUG, INFO, WARNING,
    ERROR)或错误类型,不同的处理器负责记录或处理相应的日志信息或异常。开发者可以灵活地插入或移除处理逻辑,而不影响其他日志处理。
  • 请求过滤与处理:在Web服务或API网关中,责任链可用于实现一系列预处理任务,如参数校验、请求限流、IP黑名单过滤、会话管理等,每个处理环节关注于特定的验证或转换逻辑。
  • 命令与事件处理:在处理一系列可选的或有顺序依赖的命令或事件时,责任链允许按照预定的逻辑顺序尝试执行处理逻辑,直到命令被执行或事件被妥善处理。
  • 工作流与审批流程:在企业应用中,审批流程可以设计成责任链,每个节点代表一个审批层级或角色,请求(如报销单)依次经过各个审批节点,直至最终批准或拒绝。
  • UI事件处理:在图形界面应用程序中,事件(如鼠标点击、键盘输入)可以通过责任链分发给不同的组件,每个组件决定是否消费此事件,未处理的事件继续传递给链中的下一个组件。

今天给大家的介绍的责任链有点特殊,它是基于Pipeline-Valve模型,这种模型跟常规的责任链模式有点区别

  • 每个Pipeline都是有特定的Valve,而且是在管道的最后一个执行,这个Valve叫BaseValve,并且BaseValve是不可删除的;
  • 在上层容器的管道的BaseValve中会调用下层容器的管道。

示例图:

ade76302cbfbe1d5da6e65b1bcb0e194_cb210ea701e6ba81adb4dbae15d6f022.png

如何实现Pipeline-Valve模型

1、定义Valve接口

该接口表示处理链中的单个处理单元。

public interface Valve extends Ordered {

    Valve getNextValve();

    void setNextValve(Valve next);

    void invoke(ValveContext context);

    default boolean isBaiscValve() {
        return false;
    }


}

2、定义抽象valve(可选)

注:定义该抽象valve主要是为了复用

public abstract class AbstractValve implements Valve {

    protected Valve nextValve;

    @Override
    public Valve getNextValve() {
        return nextValve;
    }

    @Override
    public void setNextValve(Valve next) {
       this.nextValve = next;
    }

    @Override
    public void invoke(ValveContext context) {
         doInvoke(context);
         if(nextValve!=null){
             nextValve.invoke(context);
         }
    }

    public abstract void doInvoke(ValveContext context);


}

3、 定义Pipeline接口

该接口主要用来用于管理Valve的集合,并提供方法来添加Valve、设置特定valve以及启动处理流程。

public interface Pipeline {

    void setBasic(Valve valve);

    void addValve(Valve valve);

    void process(ValveContext context);
}

4、定义Pipeline的默认实现

public class StandardPipeline implements Pipeline {
    /**
     * 第一个阀门
     */
    protected Valve first;

    /**
     * 最后一个阀门
     */
    protected Valve basic;


    @Override
    public void setBasic(Valve valve) {
        validateValve(valve,true);
        this.basic = valve;
    }

    @Override
    public void addValve(Valve valve) {
        validateValve(valve,false);
        if(first == null){
            this.first = valve;
            valve.setNextValve(basic);
        }else{
            Valve current = first;
            while(current != null){
               if(current.getNextValve() == basic){
                   current.setNextValve(valve);
                   valve.setNextValve(basic);
                   break;
               }
               current = current.getNextValve();
            }
        }

    }

    @Override
    public void process(ValveContext context) {
        if(first != null){
            if(context == null){
                context = new ValveContext();
            }
            first.invoke(context);
        }

    }

    public void validateValve(Valve valve,boolean isCheckBasicValve){
        Assert.notNull(valve, "valve must not be null");
        if(isCheckBasicValve){
            Assert.isTrue(valve.isBaiscValve(), "valve must be basic valve");
        }
    }
}

注: addValue else流程,是为了保证basic阀门一定是在流程最后被调用

5、定义具体valve

@Component
public class FirstValve extends AbstractValve {
    @Override
    public void doInvoke(ValveContext context) {
        String requestId = "lybgeek-" + UUID.randomUUID().toString();
        System.out.println("第一道阀门: requestId-->【" + requestId + "】");
        Map<String,Object> request = context.getRequest();
        request.put("source",FirstValve.class.getSimpleName());
        context.setRequest(request);
        context.setRequestId(requestId);
    }

    @Override
    public int getOrder() {
        return 1;
    }
}



其他阀门类似,就不列举了

6、通过Pipeline 驱动valve

public class PipelineMainTest {

    public static void main(String[] args) {
        Pipeline pipeline = new StandardPipeline();
        pipeline.setBasic(new BasicValve());
        pipeline.addValve(new FirstValve());
        pipeline.addValve(new SecondValve());
        pipeline.addValve(new ThirdValve());
        pipeline.process(new ValveContext());
    }
}
2d98e815bd09b594c7e63bfb8c715419_d4c89d88640c5f67bbc8a88edd73a93b.png

总结

如果大家对tomcat有了解的话,就会知道本文的实现其实就是tomcat的pipeline-valve的简化版实现。其次上文pipeline驱动valve的步骤可以托管给spring,文末demo链接的代码,有做了相应实现,感兴趣的朋友,可以点击文末链接查看

demo链接

https://github.com/lyb-geek/springboot-learning/tree/master/springboot-pipeline-valve

相关文章

  • 同事写了一个责任链模式,Bug无数

    什么是责任链 场景 反例 初步改造 缺点 责任链改造 责任链工厂改造 聊聊其他 最近,我让团队内一位成员写了一个导...

  • Java设计模式之责任链模式

    一、责任链模式的定义二、责任链模式的使用场景三、责任链模式UML类图四、责任链模式具体实例五、责任链模式代码实现 ...

  • 责任链模式及OkHttp中的实现

    责任链模式及OkHttp中的实现 责任链模式 责任链模式是对一个事件的处理方法,所有能对事件进行处理的对象按顺序形...

  • 2019日更挑战(八),Android-View与ViewGro

    瞎扯 上篇写了,View与ViewGroup之间的组合模式.今天就聊聊事件分发,责任链模式. 什么是责任链 举个现...

  • netty事件处理链Pipeline源码解析

    Pipeline 设计模式中有一种设计模式叫做责任链模式,netty pipeline就是责任链模式的一种实现,链...

  • 责任链模式与 OkHttp

    什么是责任链模式 OkHttp中责任链模式的实现 一、什么是责任链模式 使多个对象都有机会处理请求,从而避免了请求...

  • 责任链模式

    抽象责任链处理者 抽象责任链处理者实现类 请求抽象类或接口 请求抽象类或接口实现类 场景使用

  • 面试

    1、责任链模式 2、Bulid模式 3、Retrofit源码 (1)使用责任链模式实现 (2)Map使用 4、Re...

  • 责任链模式

    责任链模式持有一个下级对象,并在实现方法里决定处理或交给下级处理,客户提交一个请求给责任链,无需关心在哪个节点处理...

  • 设计模式之责任链模式

    责任链模式 责任链模式(Chain of Responsibility Pattern) 责任链模式将链中每一个节...

网友评论

      本文标题:聊聊如何实现一个特别的责任链

      本文链接:https://www.haomeiwen.com/subject/aynarjtx.html