大话设计模式——19.责任链模式(Chain of Responsibility Pattern)

本文介绍了职责链模式在IT中的应用,如ApacheTomcat和Springboot中的请求处理机制,以及在金融风控中的场景。该模式通过减少客户端与处理者之间的耦合,提高可维护性和扩展性,但可能导致性能和调试复杂性的问题。
摘要由CSDN通过智能技术生成

简介

使多个对象都有机会处理请求,从而避免请求的发送者和接受者之间的耦合关系。将这些对象连成一条链,并沿着这条链传递该请求,直到有一个对象处理它为止。
主要有两个核心行为:1.处理请求;2.将请求传递到下一节点

UML图:

在这里插入图片描述

应用场景
  • Apache Tomcat 对 Encoding 编码的处理,Springboot 中的拦截器和过滤器链
  • 存在多个对象可以处理同一个请求,但具体哪个请求进行处理由运行时动态决定
  • 日志级别:debugger -> info -> warning -> error

示例

金融行业的一些风控规则,受限于一些政策会经常进行调整,不同的交易金额有不同的风控策略进行处理

  1. Client:
public class Request {

    /**
     * 金额
     */
    private double money;

    /**
     * 请求类型
     */
    private String requestType;


    public double getMoney() {
        return money;
    }

    public void setMoney(double money) {
        this.money = money;
    }

    public String getRequestType() {
        return requestType;
    }

    public void setRequestType(String requestType) {
        this.requestType = requestType;
    }
}
  1. Handler:
public abstract class RiskManager {

    /**
     * 防控策略
     */
    protected String riskName;


    /**
     * 上级防控
     */
    protected RiskManager superior;

    public RiskManager(String riskName) {
        this.riskName = riskName;
    }


    /**
     * 处理请求
     *
     * @param request
     */
    public abstract void handleRequest(Request request);

    public RiskManager getSuperior() {
        return superior;
    }

    public void setSuperior(RiskManager superior) {
        this.superior = superior;
    }
}

一级防控:

public class FirstRiskManager extends RiskManager {

    public FirstRiskManager(String riskName) {
        super(riskName);
    }

    @Override
    public void handleRequest(Request request) {
        if ("线上".equals(request.getRequestType()) && 2000 > request.getMoney()) {
            System.out.println("支付方式:" + request.getRequestType() + ",支付金额:" + request.getMoney() + ",防控策略:" + riskName);
        } else {
            if (null != superior) {
                // 触发其他防控策略
                superior.handleRequest(request);
            }
        }
    }
}

二级防控:

public class SecondRiskManager extends RiskManager {

    public SecondRiskManager(String riskName) {
        super(riskName);
    }

    @Override
    public void handleRequest(Request request) {
        if ("线上".equals(request.getRequestType()) && 5000 > request.getMoney()) {
            System.out.println("支付方式:" + request.getRequestType() + ",支付金额:" + request.getMoney() + ",防控策略:" + riskName);
        } else {
            System.out.println("支付方式:" + request.getRequestType() + ",支付金额:" + request.getMoney() + ",防控策略:限制交易!");
        }
    }
}
  1. 运行
public class Main {

    public static void main(String[] args) {
        FirstRiskManager firstRiskManager = new FirstRiskManager("一级防控,正常交易");
        SecondRiskManager secondRiskManager = new SecondRiskManager("二级防控,重点监控");
        firstRiskManager.setSuperior(secondRiskManager);

        Request request = new Request();
        request.setRequestType("线上");

        // 一级防控
        request.setMoney(1200);
        firstRiskManager.handleRequest(request);

        // 二级防控
        request.setMoney(4000);
        firstRiskManager.handleRequest(request);

        // 最终防控
        request.setMoney(6000);
        firstRiskManager.handleRequest(request);
    }
}

在这里插入图片描述

总结

  • 优点
    • 客户只需要将请求发送到职责链上即可,无需关心请求的具体细节,降低了耦合度
    • 通过改变链内的调动次序,可以动态新增或删除处理类,提高了可维护性
    • 可以根据需求新增请求处理类,提高了可扩展性,同时满足开闭原则
    • 每个请求处理类只服务于自己的责任范围,满足单一职责原则
  • 缺点
    • 要把整个流程走完,可能需要很多的职责对象,可能会造成系统性能降低,同时产生大量的细粒度职责对象,并且出现问题不方便调试
    • 不能保证请求一定被接收
  • 20
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值