责任链模式
定义:
- 《设计模式》中给它的定义如下:使多个对象都有机会处理请求,从而避免请求的发送者和接收者之间的耦合关系。将这些对象连成一条链,并沿着这条链传递该请求,直到有一个对象处理它为止。
注意:责任链模式也叫职责链模式。
在责任链模式中,客户只需要将请求发送到责任链上即可,无须关心请求的处理细节和请求的传递过程,请求会自动进行传递。所以责任链将请求的发送者和请求的处理者解耦了。
组成:
- 抽象处理者(Handler)角色:
定义一个处理请求的接口,包含抽象处理方法和一个后继连接。 - 具体处理者(Concrete Handler)角色:
实现抽象处理者的处理方法,判断能否处理本次请求,如果可以处理请求则处理,否则将该请求转给它的后继者。 - 客户类(Client)角色:创建处理链:
并向链头的具体处理者对象提交请求,它不关心处理细节和请求的传递过程。
案例1
需求;
- 责任链模式,有多个对象,每个对象持有对下一个对象的引用,这样就会形成一条链,请求在这条链上传递,直到某一对象决定处理该请求.
- 但是发出者并不清楚到底最终那个对象会处理该请求,所以,责任链模式可以实现,在隐瞒客户端的情况下,对系统进行动态的调整
Abstracthandler类提供了get和set方法,方便MyHandle类设置和修改引用对象,MyHandle类是核心,实例化后生成一系列相互持有的对象,构成一条链。
抽象处理者接口: Handler
interface Handler {
public void operator();
}
抽象处理者角色: AbstractHandler
abstract class AbstractHandler {
private Handler handler;
public Handler getHandler() {
return handler;
}
public void setHandler(Handler handler) {
this.handler = handler;
}
}
具体处理者角色:MyHandler
class MyHandler extends AbstractHandler implements Handler {
private String name;
public MyHandler(String name) {
this.name = name;
}
@Override
public void operator() {
System.out.println(name+"deal!");
if(getHandler()!=null){
getHandler().operator();
}
}
}
客户类角色 创建处理链: main
public class Test01 {
public static void main(String[] args) {
MyHandler h1 = new MyHandler("h1");
MyHandler h2 = new MyHandler("h2");
MyHandler h3 = new MyHandler("h3");
h1.setHandler(h2);
h2.setHandler(h3);
h1.operator();
}
}
- 此处强调一点就是,链接上的请求可以是一条链,可以是一个树,还可以是一个环,模式本身不约束这个,需要我们自己去实现,
- 同时,在一个时刻,命令只允许由一个对象传给另一个对象,而不允许传给多个对象。
案例2 :模拟servlet 过滤器 Filter
需求: 对信息进行过滤.
- 对网络传过来的请求request,和服务器传过来的response都要实施过滤
- 对request要求过滤器按添加顺序,过滤器1,过滤器2进行过滤.
- 对response要求按过滤器2,过滤器1进行过滤.
方法1 :
import java.util.ArrayList;
import java.util.List;
public class Servlet_Main01 {
public static void main(String[] args) {
Request request = new Request();
request.str = "大家好:),<script>,欢迎访问 mashibing.com ,大家都是996 ";
Response response = new Response();
response.str = "response";
FilterChain chain = new FilterChain();
chain.add(new HTMLFilter()).add(new SensitiveFilter());
chain.doFilter(request, response);
System.out.println(request.str);
System.out.println(response.str);
}
}
interface Filter {
boolean doFilter(Request request, Response response);
}
class HTMLFilter implements Filter {
@Override
public boolean doFilter(Request request, Response response) {
request.str = request.str.replaceAll("<", "[").replaceAll(">", "]");
response.str += "--HTMLFilter()";
return true;
}
}
class Request {
String str;
}
class Response {
String str;
}
class SensitiveFilter implements Filter {
@Override
public boolean doFilter(Request request, Response response) {
request.str = request.str.replaceAll("996", "955");
response.str += "--SensitiveFilter()";
return true;
}
}
class FilterChain implements Filter {
List<Filter> filters = new ArrayList<>();
public FilterChain add(Filter f) {
filters.add(f);
return this;
}
public boolean doFilter(Request request, Response response) {
for(Filter f : filters ){
f.doFilter(request, response);
}
return true;
}
}
结果;
- request : 先HTMLFilter ,后SensitiveFilter
- response: 先HTMLFilter ,后SensitiveFilter
方法2 :
在filterChain中处理加入位置的记录,同时在filter中加入第三个参数
import java.util.ArrayList;
import java.util.List;
public class Servlet_Main {
public static void main(String[] args) {
Request request = new Request();
request.str = "大家好:),<script>,欢迎访问 mashibing.com ,大家都是996 ";
Response response = new Response();
response.str = "response";
FilterChain chain = new FilterChain();
chain.add(new HTMLFilter()).add(new SensitiveFilter());
chain.doFilter(request, response);
System.out.println(request.str);
System.out.println(response.str);
}
}
interface Filter {
void doFilter(Request request, Response response, FilterChain chain);
}
class HTMLFilter implements Filter {
@Override
public void doFilter(Request request, Response response, FilterChain chain) {
request.str = request.str.replaceAll("<", "[").replaceAll(">", "]") + "HTMLFilter()";
chain.doFilter(request, response);
response.str += "--HTMLFilter()";
}
}
class Request {
String str;
}
class Response {
String str;
}
class SensitiveFilter implements Filter {
@Override
public void doFilter(Request request, Response response, FilterChain chain) {
request.str = request.str.replaceAll("996", "955") + " SensitiveFilter()";
chain.doFilter(request, response);
response.str += "--SensitiveFilter()";
}
}
class FilterChain {
List<Filter> filters = new ArrayList<>();
int index = 0;
public FilterChain add(Filter f) {
filters.add(f);
return this;
}
public void doFilter(Request request, Response response) {
if(index == filters.size()) return;
Filter f = filters.get(index);
index ++;
f.doFilter(request, response, this);
}
}
结果;
- request : 先HTMLFilter ,后SensitiveFilter
- response: 先SensitiveFilter,后HTMLFilter
- 符合需求
servlet 过滤器源码
https://docs.oracle.com/javaee/7/api/toc.htm
Filter
FilterChain
Spring的实现MockFilterChain:
- 从上面两段代码可以发现,子类通过一个List来构建“链路”,最终调用的时候就是通过遍历List来实现“链路”传递。
总结:
责任链模式存在以下两种情况:
-
纯的职责链模式:
一个请求必须被某一个处理者对象所接收,且一个具体处理者对某个请求的处理只能采用以下两种行为之一:自己处理(承担责任);把责任推给下家处理。 -
不纯的职责链模式:
允许出现某一个具体处理者对象在承担了请求的一部分责任后又将剩余的责任传给下家的情况,且一个请求可以最终不被任何接收端对象所接收。
责任链模式 适用场景:
责任链模式主要是解耦了请求与处理,用户只需要将请求发送到链上即可,无需关心请求的具体内容和处理细节,请求会自动进行传递直至有节点进行处理。可以适用于如下场景:
- 多个对象可以处理同一请求,但具体由哪个对象处理则在运行时动态决定。
- 在不明确指定接收者的情况下,向多个对象中的一个提交请求
- 可以动态指定一组对象的处理请求。
优点:
- 降低耦合度。它将请求的发送者和接收者解耦。
- 简化了对象。使得对象不需要知道链的结构。
- 增强给对象指派职责的灵活性。通过改变链内的成员或者调动它们的次序,允许动态地新增或者删除责任。
- 增加新的请求处理类很方便。
缺点:
- 如果责任链的链路太长或者处理时间过程,会影响性能。
- 如果节点对象存在循环引用时,会造成死循环,导致系统崩溃