设计模式-职责链模式

news/2024/7/7 22:12:19

职责链模式


文章目录

  • 职责链模式
  • 什么是职责链模式
  • 为什么要用职责链模式
  • 如何使用职责链模式
  • 总结


什么是职责链模式

  将请求的发送和接收解耦,让多个接收对象都有机会处理这个请求。将这些接收对象串成一条链,并沿着这条链传递这个请求,直到链上的某个接收对象能够处理它为止。
  这么说比较抽象,我用更加容易理解的话来进一步解读一下。
  在职责链模式中,多个处理器(也就是刚刚定义中说的“接收对象”)依次处理同一个请求。一个请求先经过 A 处理器处理,然后再把请求传递给 B 处理器,B 处理器处理完后再传递给 C 处理器,以此类推,形成一个链条。链条上的每个处理器各自承担各自的处理职责,所以叫作职责链模式。


为什么要用职责链模式

  应用设计模式主要是为了应对代码的复杂性,让其满足开闭原则,提高代码的扩展性。这里应用职责链模式也不例外。
  我们举例来说明,现在我们有个这样的需求,我们现在的系统分为管理端和终端,管理端可以管控终端软件的安装卸载等,可以在管理端上传安装包,然后下发到终端安装。上传安装包时需要校验安装包是否有病毒:root权限获取、木马、流氓广告弹窗等;使用职责链模式之前的代码可能是这样的:

public class InstallationPackageVirusFilter {
    public boolean filter(InstallationPackageInfo installationPackageInfo) {
        if (!filterRootePrmissioncAquisition(installationPackageInfo)) {
            return false;
        }
        if (!filterTrojanHorse(installationPackageInfo)) {
            return false;
        }
        if (!filterAdvertisingPopUp(installationPackageInfo)) {
            return false;
        }
        return true;
    }

    private boolean filterRootPermissionAcquisition(InstallationPackageInfo installationPackageInfo) {
        // root权限获取
    }

    private boolean filterTrojanHorse(InstallationPackageInfo installationPackageInfo) {
        // 木马
    }

    private boolean filterAdvertisingPopUp(InstallationPackageInfo installationPackageInfo) {
        // 过滤广告弹框
    }
}

  上面这段代码用起来没有问题,那么我们如果需要增加一种病毒类型的检测,那么就需要增加一个if分支并在此类中再增加一个工具方法,这显然是不复合开闭原则的。
  接下来,我们使用职责链模式对这段代码进行优化。


如何使用职责链模式

  职责链模式有两种实现方式,一种是链表形式,一种是数组形式,接下来我们给出两种形式的代码示例。

  • 链表形式的职责链模式
// 安装包病毒过滤器抽象类
public abstract class InstallationPackageVirusFilter {
    // 下一过滤器
    protected InstallationPackageVirusFilter nextFilter = null;

    // 设置下一过滤器
    public void setNextFilter(InstallationPackageVirusFilter nextFilter) {
        this.nextFilter = nextFilter;
    }

    // 执行处理(此处使用模板模式,避免重写时忘记执行nextFilter.handle())
    public final void handle() {
        boolean handled = doHandle();
        if (nextFilter != null && !handled) {
            nextFilter.handle();
        }
    }

    // 具体逻辑
    protected abstract boolean doHandle();
}

// Root权限获取过滤器
public class RootPermissionAcquireFilter extends InstallationPackageVirusFilter{
    @Override
    protected boolean doHandle() {
        // 具体的检测逻辑...
        return false;
    }
}

// 此处省略其他具象类的实现,与Root权限获取过滤器一致
...

// 安装包病毒过滤器链
public class InstallationPackageVirusFilterChain {
    // 链表的头
    private InstallationPackageVirusFilter head = null;
    // 链表的尾
    private InstallationPackageVirusFilter tail = null;

    // 添加过滤器
    public void addHandler(InstallationPackageVirusFilter handler) {
        handler.setNextFilter(null);
        if (head == null) {
            head = handler;
            tail = handler;
            return;
        }
        tail.setNextFilter(handler);
        tail = handler;
    }

    // 执行过滤逻辑
    public void handle() {
        if (head != null) {
            head.handle();
        }
    }
}

// 具体应用
public class Test {
    public static void main(String[] args) {
        InstallationPackageVirusFilterChain chain = new InstallationPackageVirusFilterChain();
        chain.addHandler(new RootPermissionAcquireFilter());
        // 省略其他过滤器的添加
        chain.addHandler(...);
        chain.handle();
    }
}
  • 数组形式的职责链模式
// 安装包病毒过滤器接口
public interface IInstallationPackageVirusFilter {
    boolean handle();
}


// Root权限获取过滤器
public class RootPermissionAcquireFilter implements IInstallationPackageVirusFilter{
    @Override
    public boolean handle() {
        // 具体的检测逻辑...
        return false;
    }
}

// 此处省略其他具象类的实现,与Root权限获取过滤器一致
...

// 安装包病毒过滤器链
public class InstallationPackageVirusFilterChain {
    private final List<IInstallationPackageVirusFilter> handlers = new ArrayList<>();

    // 添加过滤器
    public void addHandler(IInstallationPackageVirusFilter handler) {
        this.handlers.add(handler);
    }

    // 处理
    public void handle() {
        for (IInstallationPackageVirusFilter handler : handlers) {
            boolean handled = handler.handle();
            if (handled) {
                break;
            }
        }
    }
}

// 具体应用
public class Test {
    public static void main(String[] args) {
        chain.chain.InstallationPackageVirusFilterChain chain = new InstallationPackageVirusFilterChain();
        chain.addHandler(new RootPermissionAcquireFilter());
        // 省略其他过滤器的添加
        // chain.addHandler(...);
        chain.handle();
    }
}

  通过上面的代码,我们可以明显感觉到比使用职责链之前的代码简洁,增加一种病毒检测只需要继承/实现指定的抽象类或接口,在使用的时候添加到链中即可。


总结

  • 职责链模式可以应对代码的复杂性。
      将大块代码逻辑拆分成函数,将大类拆分成小类,是应对代码复杂性的常用方法。应用职责链模式,我们把各个安装包病毒过滤函数拆分出来,设计成独立的类,进一步简化了 InstallationPackageVirusFilter类,让 InstallationPackageVirusFilter 类的代码不会过多,过复杂。
  • 职责链模式更满足开闭原则,提高代码的扩展性。
      当我们要扩展新的过滤算法的时候,比如,我们还需要过滤扣费,按照非职责链模式的代码实现方式,我们需要修改 InstallationPackageVirusFilter 的代码,违反开闭原则。不过,这样的修改还算比较集中,也是可以接受的。
      而职责链模式的实现方式更加优雅,只需要新添加一个 Filter 类,并且通过 addFilter() 函数将它添加到 FilterChain 中即可,其他代码完全不需要修改。 不过,你可能会说,即便使用职责链模式来实现,当添加新的过滤算法的时候,还是要修改客户端代码(Test),这样做也没有完全符合开闭原则。
      实际上,细化一下的话,我们可以把上面的代码分成两类:框架代码和客户端代码。其中,Test 属于客户端代码,也就是使用框架的代码。除 Test 之外的代码属于敏感词过滤框架代码。 假设敏感词过滤框架并不是我们开发维护的,而是我们引入的一个第三方框架,我们要扩展一个新的过滤算法,不可能直接去修改框架的源码。这个时候,利用职责链模式就能达到开篇所说的,在不修改框架源码的情况下,基于职责链模式提供的扩展点,来扩展新的功能。换句话说,我们在框架这个代码范围内实现了开闭原则。
      除此之外,利用职责链模式相对于不用职责链的实现方式,还有一个好处,那就是配置过滤算法更加灵活,可以只选择使用某几个过滤算法。

http://lihuaxi.xjx100.cn/news/1133163.html

相关文章

定风波、渡重山、至未来:2023中国数字能源生态大会开启的新旅程

全球碳中和的时代背景下&#xff0c;面向3060发展目标&#xff0c;新使命、新技术、新应用的到来&#xff0c;都给能源产业带来了持续变革的必要性与可能性。 2023年5月11日&#xff0c;在2023中国数字能源生态大会上&#xff0c;华为数字能源技术有限公司总裁侯金龙发表了“融…

如何使用SVG.js库中的mask()和clip()方法创建这两种元素,以及它们的常见属性和用法

SVG.js是一款轻量级的JavaScript库&#xff0c;用于创建交互式的可缩放矢量图形&#xff08;SVG&#xff09;动画。在SVG.js中&#xff0c;mask和clipPath元素是非常有用的功能&#xff0c;能够将SVG图像切割、裁剪并添加效果。本文将为您介绍SVG.js中的mask和clipPath元素的相…

QML APP开发套路(三):前/后端值传递(自定义值类型)

&#xff08;1&#xff09;前/后端交互内容 QML APP前后端交互的内容按目标&#xff08;拍脑袋&#xff09;可以分为2个部分&#xff1a; 方向内容后端&#xff08;C&#xff09; → 前端&#xff08;QML&#xff09;前端展示所需的数据&#xff0c;形式&#xff1a;简单类型…

电脑技巧:分享六个有趣好玩的网站,值得收藏

目录 1、Weavesilk 2、一键抠图 3、狗屁不通文章生成器 4、小霸王在线小游戏 5、世界名画在线拼图 6、寻找不动的emoji 今天小编给大家分享六个有趣好玩的网站&#xff0c;值得收藏&#xff01; 1、Weavesilk Weavesilk是一个光线绘画网站&#xff0c;它不需要有任何绘画…

Linux -- 进程信号

文章目录 1. 信号量1.1 进程互斥概念1.2 认识信号量 2. 信号入门2.1 信号概念2.2 见一见2.3 signal()系统调用2.4 宏定义信号 3. 信号产生方式3.1 键盘产生信号3.2 系统调用产生信号3.3 软件条件产生信号3.4 硬件异常产生信号3.5 Core dump 4. 阻塞信号4.1 相关概念4.2 信号在内…

《程序员面试金典(第6版)》面试题 16.22. 兰顿蚂蚁(哈希映射,C++)

题目描述 一只蚂蚁坐在由白色和黑色方格构成的无限网格上。开始时&#xff0c;网格全白&#xff0c;蚂蚁面向右侧。每行走一步&#xff0c;蚂蚁执行以下操作。传送门 (1) 如果在白色方格上&#xff0c;则翻转方格的颜色&#xff0c;向右(顺时针)转 90 度&#xff0c;并向前移动…

解密Netty中的Reactor模式

文章目录 单线程Reactor模式多线程Reactor模式Reactor模式中IO事件的处理流程Netty中的通道ChannelNetty中的反应器ReactorNetty中的处理器HandlerNetty中的通道Channel和处理器Handler的协作组件Pipeline Reactor(反应器)模式是高性能网络编程在设计和架构方面的基础模式.Doug…

基于qt5的应用程序在windows和linux环境下修改图标及制定后缀关联

基于qt5的应用程序在windows和linux环境下修改图标及制定后缀关联 1、windows 1.1 修改应用程序图标 方式一&#xff1a; 使用qmake来生成makefile文件&#xff0c;只需要在.pro中添加&#xff1a; RC_ICONS logo.ico 然后&#xff0c;重新生成makefile文件和应用程序&…