很多人觉得自己写的是业务代码,按照逻辑写下去,再把公用的方法抽出来复用就可以了,设计模式根本就没必要用,更没必要学。

一开始的时候,我也是这么想,直到我遇到。。。

举个例子

我们先看一个普通的下单拦截接口。

基本逻辑,参数安全拦截,次数拦截,规则拦截,都通过,返回允许下单,任意一个失败,返回对应的失败原因。

多层嵌套if写法

我们正常多层嵌套if的写法

  1. /**
     * @author saier
     * @date 2020/3/31 18:03
     */
    public class Order {
        public Message interrupt1(){
            return null;
        }
        public Message interrupt2(){
            return null;
        }
        public Message interrupt3(){
            return null;
        }
        public Message interrupt4(){
            return null;
        }
        public Message interrupt5(){
            return null;
        }

        public static void main(String[] args) {
            Order order= new Order();
            if(order.interrupt1().getResult() == 1){
                if(order.interrupt2().getResult() == 1){
                    if(order.interrupt3().getResult() == 1){
                        if(order.interrupt4().getResult() == 1){
                            if(order.interrupt5().getResult() == 1){
                                System.out.println("success");
                            }
                        }
                    }
                }
            }

        }
    }

    @Data
    class Message {
        private int result;
        private String msg;
    }

异常处理逻辑

或者有些利用异常做逻辑,代码会简单一点

  1. /**
     * @author saier
     * @date 2020/3/31 18:03
     */
    public class Order2 {
        public void interrupt1(){

        }
        public void interrupt2(){

        }
        public void interrupt3(){
            //失败
            throw new RuntimeException();
        }
        public void interrupt4(){
            //失败
            throw new RuntimeException();
        }
        public void interrupt5(){
            //失败
            throw new RuntimeException();
        }

        public static void main(String[] args) {
            Order2 order2= new Order2();
            try{
                order2.interrupt1();
                order2.interrupt2();
                order2.interrupt3();
                order2.interrupt4();
                order2.interrupt5();
                System.out.println("success");
            }catch (RuntimeException e){
                System.out.println("fail");
            }

        }
    }

一开始,我就直接使用异常来做逻辑。但后续逻辑越来越复杂之后,也会出现一些问题。例如异常只能返回异常信息,不能返回更多的字段信息。

后面也留意到,异常做逻辑,在阿里规范是禁止的。

阿里代码规范 :
【强制】异常不要用来做流程控制,条件控制。
说明:异常设计的初衷是解决程序运行中的各种意外情况,且异常的处理效率比条件判断方式要低很多。

更重要的是,代码可读性太差了,随时一个方法的异常抛出来,还要考虑代码本身的异常。

没更好的办法,只能考虑设计模式了

怎么改,会使代码的可读性高,扩展性好?

在同事的提醒下,突然想起了设计模式!

我们希望达到的目的

  1. 代码没有这么多if else嵌套,可读性高
  2. 如果新增新的拦截逻辑简单方便,不影响原本的逻辑,扩展性好
  3. 可以很方便地调换拦截逻辑顺序,低耦合

责任链模式

在这种场景下,非常适合责任链模式。(什么场景使用什么设计模式,这就需要平时有积累,知道各种设计模式的基本使用)

责任链,顾名思义,就是用来处理相关事务责任的一条执行链,执行链上有多个节点,每个节点都有机会(条件匹配)处理请求事务,如果某个节点处理完了就可以根据实际业务需求传递给下一个节点继续处理或者返回处理完毕。

首先,建立过滤器的抽象类

  1. public abstract class AbstractFilter {

        private AbstractFilter nextFilter;

        /**
         * 责任链的下一个元素
         */
        public void setNextFilter(AbstractFilter nextFilter){
            this.nextFilter = nextFilter;
        }

        public AbstractFilter getLastFilter(){
            if(this.nextFilter != null){
                return this.nextFilter.getLastFilter();
            }else{
                return this;
            }
        }

        public void filter(FilterRequest filterRequest, Response response){
            doFilter(filterRequest,response);
            if(response.isFilterNext() && nextFilter != null){
                nextFilter.filter(filterRequest,response);
            }
        }

        /**
         * 具体拦截逻辑
         */
        public abstract void doFilter(FilterRequest filterRequest, Response response);

        /**
         * 根据拦截结果做处理
         */
        public void exec(FilterRequest filterRequest, Response response){
        }
    }

过滤器的实现类

  1. @Component
    @Order(5)
    public class CheckParamFilter1 extends AbstractFilter {
        @Override
        public void doFilter(FilterRequest filterRequest, Response response) {

        }
    }

    @Component
    @Order(10)
    public class CheckParamFilter2 extends AbstractFilter {
        @Override
        public void doFilter(FilterRequest filterRequest, Response response) {

        }
    }

使用Order注解,确定过滤器的顺序,后续在spring注入的时候,会有奇效

  1. //利用spring的自动注入机制
    @Autowired
    List<AbstractFilter> abstractFilterList;

    private AbstractFilter firstFilter;

    //spring注入后自动执行
    @PostConstruct
    public void initializeChainFilter(){
        //把所有调用的逻辑注入到责任链,按照Order排序,越小优先级越高
        for(int i = 0;i<abstractFilterList.size();i++){
            if(== 0){
                firstFilter = abstractFilterList.get(i);
            }else{
                firstFilter.getLastFilter().setNextFilter(abstractFilterList.get(i));
            }
        }
    }

    //直接使用
    public Response exec(){
        firstFilter.filter(filterRequest, response);
        return response;
    }

使用设计模式的好处

看下使用责任链模式后,有什么好处!

  1. 新增拦截逻辑,只需要再实现一个AbstractFilter类即可
  2. 修改拦截顺序,只需要修改Order注解的大小,越小,优先级越高
  3. 代码清晰,所有处理逻辑下沉到实现类中

使用设计模式的缺点

做到了低耦合,高扩展。但也带来了一些不好的地方

  1. 逻辑更复杂,用了链式等数据结构,要注意单例的问题,不能重复使用
  2. 类数量激增,一个拦截器就一个类

最后小结一下

不是什么地方都适合使用设计模式,如果逻辑简单,你硬要使用设计模式,只会带来结构上的复杂,大家可以按照大家的业务场景来使用。

谁能干掉了if else的更多相关文章

  1. 干掉命令行窗口下MySql乱码

    晚上重温dos窗口操作mysql的时候,遇到了一个巨蛋疼的问题------>中文验证码  -->_-->,所以找了找资料弄懂了怎么解决乱码问题,,小记一下. 新建一个表 create ...

  2. 干掉Unity3D

    我为什么想干掉Unity3D? 这个问题容我不答,每个做技术的人总有一些完美主义. 你使用u3d的过程中是不是痛并快乐着呢. 就从两个国内具有相当普遍性的痛点说起. il2cpp,unity作出了这个 ...

  3. Markdown会干掉Html吗?

    Markdown会干掉Html吗? 很明显,MarkDown正在已一种比病毒还快的速度传播着,量子的机器人语言也是深受其启发,当然了,在这个东西没搞出来之前,MarkDown就能干很多事情,比如在线编 ...

  4. Excel应该这么玩——1、命名单元格:干掉常数

    命名单元格:通过名称来引用单元格中的值,常用于引用固定不变的值. 单元格是Excel中存储数据的最小单位,在公式中通过A1.B2之类的名称来引用其中的值.A1只是单元格的坐标,就好像人的身份证号.生活 ...

  5. 夺命雷公狗---DEDECMS----8dedecms干掉首页和-文档页-栏目页的页面的广告

    我们首先来将首页生成静态页面,如下图所示: 成功后,如下显示: 如果成功后则在文件夹下多了一个index.html的文件.. 我们的首页静态页面是通过模版文件生成,所以我们只需要把模版文件的广告标签删 ...

  6. 干掉cmd:windows下使用linux命令行

    对于喜欢用命令行的朋友们,在windows下面使用cmd窗口是不是很不爽?复制不方便?不能随意放大缩小?如果需要多个控制台要多个窗口?....各种不爽 一.基础工具 如果你也不爽,那就对了,所以给大家 ...

  7. Linux中 干掉原来的PHP方法

    干掉原来的PHP方法: 查看php版本命令:#php -v这个命令是删除不干净的#yum remove php因为使用这个命令以后再用#php -v还是会看到有版本信息的..... 必须强制删除#rp ...

  8. Xcode6为什么干掉pch(Precompile Prefix Header)&如何添加pch文件

    转载:  http://blog.csdn.net/iosdevtip/article/details/40918353 一直在用xcode6开发,但项目都是在xcode5上创建的,所以一直没注意到, ...

  9. Xcode6为什么干掉pch(Precompile Prefix Header)&amp;怎样加入pch文件

    一直在用xcode6开发,但项目都是在xcode5上创建的,所以一直没注意到,xcode6居然干掉pch文件了. 为什么xcode6没有自己主动创建pch文件呢? 简单地看:我们在写项目的时候,大部分 ...

  10. 使用gc、objgraph干掉python内存泄露与循环引用!

    Python使用引用计数和垃圾回收来做内存管理,前面也写过一遍文章<Python内存优化>,介绍了在python中,如何profile内存使用情况,并做出相应的优化.本文介绍两个更致命的问 ...

随机推荐

  1. MySQL如何快速插入数据

    前言: 日常学习和工作中,经常会遇到导数据的需求.比如数据迁移.数据恢复.新建从库等,这些操作可能都会涉及大量数据的导入.有时候导入进度慢,电脑风扇狂转真的很让人崩溃,其实有些小技巧是可以让导入更快速 ...

  2. OOP第四章博客

    OOP第四章博客作业 (1)本单元作业架构设计 1)针对于第一次作业,我是将所给类进行了自己的封装,在MyUmlInteraction类里面进行关系的建立,这里把所给的UmlClass建立好,同时有i ...

  3. Pytorch_Part7_模型使用

    VisualPytorch beta发布了! 功能概述:通过可视化拖拽网络层方式搭建模型,可选择不同数据集.损失函数.优化器生成可运行pytorch代码 扩展功能:1. 模型搭建支持模块的嵌套:2. ...

  4. Excel导出数据Excel.Application组件权限设置方法

    很多网络应用系统都会涉及到数据采用Excel方式导出的模块,部分朋友问我到底怎么弄,其实方式很多种,目前比较优秀的方式还是直接用Excel的Excel.Application方式比较合适. 采用Exc ...

  5. [bug] MapReduce卡死

    参考 https://blog.csdn.net/WYpersist/article/details/80202055

  6. [刷题] PTA 02-线性结构4 Pop Sequence

    模拟栈进出 方法一: 1 #include<stdio.h> 2 #define MAXSIZE 1000 3 4 typedef struct{ 5 int data[MAXSIZE]; ...

  7. 【海通国际】Joe Lowry(Mr. Lithium)谈全球电池原材料供应危机

    [海通国际]Joe Lowry(Mr. Lithium)谈全球电池原材料供应危机 环球锂业公司(Global Lithium)总裁Joe Lowry日前接受了欧洲锰业Euro Manganese的邀请 ...

  8. unrar命令解压rar unrar e XXX.rar (验证通过20200511)

    unrar命令解压rar 一个从入门到放弃再到改行的工程师 2018-05-02 17:53:04 3916 收藏展开压缩tar -cvf jpg.tar *.jpg //将目录里所有jpg文件打包成 ...

  9. 【山外笔记-SVN命令】svn命令详解

    本文打印版文件下载地址 [山外笔记-SVN命令]svn命令详解-打印版.pdf 一.命令简介 svn命令用于Subversion命令行客户端,执行svn相关的操作. 二.命令语法 1.svn语法: ( ...

  10. 搭建LAMP环境部署Ecshop电商网站

    实战-部署Ecshop电商网站 实验环境 Centos7 ip:192.168.121.17 一.关闭防火墙和selinux [root@localhost ~]# systemctl stop fi ...