责任链的2种实现方式,你更pick哪一种

 戳蓝字「TopCoder」关注我们哦!

责任链的2种实现方式,你更pick哪一种_第1张图片

编者注:责任链模式是日常开发或者框架中经常被使用的一种设计模式,典型的责任链有2种实现方式,不知道各位小伙伴更pick哪一种呢?下面就一起来比较下这2种实现方式吧~

1 责任链是什么

责任链是属于行为型模式,在这种模式中,通常每个接收者都包含对另一个接收者的引用,如果一个对象不能处理该请求,那么它会把相同的请求传给下一个接收者,依此类推。责任链模式避免请求发送者与接收者耦合在一起,让多个对象都有可能接收请求,将这些对象连接成一条链,并且沿着这条链传递请求,直到有对象处理它为止。

责任链类图下图:

责任链的2种实现方式,你更pick哪一种_第2张图片

责任链模式在开源项目中应用场景还是比较常见的,比如Tomcat中的Filter处理链、Netty中的ChannelHandler处理链、Dubbo RPC中的consumer侧的Filter链等等。责任链模式应用在业务流程中的 多个同类型操作场景,相当于对一个复杂较长的操作进行分段处理,这样对扩展性友好,新增操作阶段时更加灵活。这种可以理解为分片思想,降低业务流程操作的复杂度。

2 责任链的2种实现

常见的责任链流程如下:

责任链的2种实现方式,你更pick哪一种_第3张图片

2.1 节点传递方式

节点传递方式也就是,责任链中当前节点处理完成之后,自己传递给下一个处理节点继续处理。

public interface Handler {
    default boolean match(String msg) {
        return true;
    }
    void process(String msg);
}

public abstract class AbstractHandler implements Handler {
    private Handler next;

    public AbstractHandler setNextHandler(Handler next) {
        this.next = next;
        return this;
    }

    @Override
    public void process(String msg) {
        doProcess(msg);

        if (next != null) {
            next.process(msg);
        }
    }

    protected abstract void doProcess(String msg);
}

// 具体的责任链处理器
public class Handler1 extends AbstractHandler {
    @Override
    public void doProcess(String msg) {
        System.out.println("[Handler1] process " + msg);
    }
}
public class Handler2 extends AbstractHandler {
    @Override
    protected void doProcess(String msg) {
        System.out.println("[Handler2] process " + msg);
    }
}
public class Handler3 extends AbstractHandler {
    @Override
    protected void doProcess(String msg) {
        System.out.println("[Handler3] process " + msg);
    }
}

输出结果:

责任链的2种实现方式,你更pick哪一种_第4张图片

2.2 统一传递方式

统一传递方式也就是,不由责任链中处理节点传递给下一个节点,而是由统一的传递逻辑进行传递。

public class HandlerWrap {
    private List handlerList = new ArrayList<>();

    public HandlerWrap() {
        handlerList.add(new Handler1());
        handlerList.add(new Handler2());
        handlerList.add(new Handler3());
    }

    public void process(String msg) {
        for (Handler handler : handlerList) {
            handler.process(msg);
        }
    }
}

public class Handler1 implements Handler {
    @Override
    public void process(String msg) {
        System.out.println("[Handler1] process " + msg);
    }
}
public class Handler2 implements Handler {
    @Override
    public void process(String msg) {
        System.out.println("[Handler2] process " + msg);
    }
}
public class Handler3 implements Handler {
    @Override
    public void process(String msg) {
        System.out.println("[Handler3] process " + msg);
    }
}

输出结果:

责任链的2种实现方式,你更pick哪一种_第5张图片

3 两种实现方式的比较

上述两种实现方式差别就是谁来进行下一个节点的传递工作,节点传递方式 是责任链中当前处理节点处理完成之后,自己传递给下一个节点;统一传递方式 是在统一的地方进行传递工作,减轻处理节点的“负担”。

二者本质上是一样的,不过前一种实现方式初始化成本较高,还要注意处理节点的前后顺序,这种调整一个节点的位置时特别要注意前后节点的关系,否则处理链顺序就错乱了。

后续开发中,建议使用第二种实现方式,这种责任链初始化成本较低,调整责任链成本较小。不过有些责任链使用场景中,会将前一个处理节点的返回结果作为下一个处理节点的入参,这种场景一般推荐使用第一种实现方式,就像Netty中的ChannelHandler处理链流程类似。

如果小伙伴们还知道其他实现方式或者更多的应用场景,欢迎留言讨论哈~

 推荐阅读 

  • 从socket api看网络通信流程

  • 深入理解TCP 3次握手

  • 谈一谈TCP的4次挥手

  • Linux信号处理机制

  • Linux管道那些事儿


欢迎小伙伴关注【TopCoder】阅读更多精彩好文。

你可能感兴趣的:(责任链的2种实现方式,你更pick哪一种)