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

责任链是什么

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

责任链类图下图:

image-20200620215443800

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

责任链的2种实现

常见的责任链流程如下:

image-20200620215503303

节点传递方式

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

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
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);
}
}

输出结果:

image-20200620215550879

统一传递方式

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

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
public class HandlerWrap {
private List<Handler> 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);
}
}

输出结果:

image-20200620215623531

两种实现方式的比较

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

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

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