Loading [MathJax]/jax/output/CommonHTML/config.js
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >专栏 >设计模式都没用过,好意思出去面试?

设计模式都没用过,好意思出去面试?

作者头像
Java技术栈
发布于 2019-10-14 08:55:27
发布于 2019-10-14 08:55:27
54600
代码可运行
举报
文章被收录于专栏:Java技术栈Java技术栈
运行总次数:0
代码可运行
为什么要使用设计模式

因为我们的项目的需求是永远在变的,为了应对这种变化,使得我们的代码能够轻易的实现解耦和拓展。如果能够保证代码一次写好以后都不会再改变了,那可以想怎么写怎么写了。

如何判断那里需要使用设计模式

在我们实现中,有一些代码是一次写好后续基本不会改变的,或者不太需要扩展的,比如一些工具类等。有一部分是会经常变得,设计模式大多都应用在需求会变化的这一部分。分析这些代码会如何变,选择合适的设计模式来优化这部分代码。

以促销活动需求为例

需求

为了促进商品的销售,各大电商品台会在平时或者一些节日的时候退出一些促销活动刺激用户消费,活动的类型可能会各不相同,如下:

  • 满减,满400减20
  • 代金卷,玛莎拉蒂5元代金卷
  • 折扣,9折,8折
  • 每满减,每满200减10
  • 等等

其中有些可以叠加,有些只能单独使用。

简单实现

上面的需求看起来还是比较简单的,但是如果考虑到我们是不可能一次定义好所有的促销活动类型,后续我们可能会随时都添加新的类型,要保证能够简单的实现功能扩展,那就比较麻烦了。Spring 框架用到的 9 个设计模式汇总,这个你知道吗?

先拿到需求的时候,也不用去想那么多,挽起袖子就是一通操作:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
public class OrderPromotion {

    public BigDecimal promotion(Order order, int[] promotions){
        for(int promotion:promotions){
            switch (promotion){
                case 1:
                    //计算该类型折扣后的价格
                    break;
                case 2:
                    //计算该类型折扣后的价格
                    break;
                case 3:
                    //计算该类型折扣后的价格
                    break;
                //....
            }
        }
        return order.getResultPrice();
    }
}

单从功能实现上来说,上面的代码已经完成了基本功能了。

但是上面的代码也是致命的,虽然看起来很简单,但是那只不过是因为大多数功能都用注释代替了,换成实际代码的话一个方法可能就得上千行。

尤其是当我们需要添加新的促销活动的话就需要在switch中添加新的类型,这对于开发来说简直是灾难,并且维护这些代码也是一个麻烦。

优化一:单一职责原则

上面的代码中,promotion(…)方法直接完成了所有的工作,但是咋我们实际实现中最好让一个方法的职责单一,只完成某一个功能,所以这里我们将对折扣类型的判断和计算价格分开:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
public class OrderPromotion {

    public BigDecimal promotion(Order order, int[] promotions){
        for(int promotion:promotions){
            switch (promotion){
                case 1:
                    calculate1(order);
                    break;
                case 2:
                    calculate2(order);
                    break;
                case 3:
                    calculate3(order);
                    break;
                //more promotion
            }
        }
        return order.getResultPrice();
    }

    public void calculate1(Order order){
        //计算使用折扣一后的价格
    }

    public void calculate2(Order order){
        //计算使用折扣二后的价格
    }

    public void calculate3(Order order){
        //计算使用折扣三后的价格
    }

    //more calculate

}

这里我们将折扣类型的判断和计算价格分开,使得promotion(…)方法的代码量大大降低,提升了代码的可读性。面象对象设计6大原则之一:单一职责原则,这篇也推荐大家看下。

优化二:策略模式

上面优化后的代码提升了原有代码的可读性,但是原来OrderPromotion类代码大爆炸的问题还是没有解决。

针对这个问题,我们希望能够将计算的代码和当前代码分离开,首先我们能想到的就是定义一个类,然后将计算的代码复制到这个类中,需要的时候就调用。这样到的确是分离开了,但是完全是治标不治本。在添加新的促销活动是两个类都要改。

所以我们希望能够将不同的促销活动的实现分离开,这样对每一种活动的实现都是分开的,修改也不会影响其他的,基于此我们完全可以选择策略模式来实现。

策略模式

策略模式的思想是针对一组算法,将每一种算法都封装到具有共同接口的独立的类中,从而是它们可以相互替换。策略模式的最大特点是使得算法可以在不影响客户端的情况下发生变化,从而改变不同的功能。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
public class OrderPromotion {

    public BigDecimal promotion(Order order, int[] promotions){
        for(int promotion:promotions){
            switch (promotion){
                case 1:
                    new PromotionType1Calculate(order);
                    break;
                case 2:
                    new PromotionType1Calculate(order);
                    break;
                case 3:
                    new PromotionType1Calculate(order);
                    break;
                //more promotion
            }
        }
        return order.getResultPrice();
    }
}

上面的代码很明显已经精简很多了,到了现在如果需要添加一个促销活动的话只需定义一个促销类,实现PromotionCalculation接口然后在switch中添加即可。

优化三:工厂模式

上面的代码虽然已经将促销活动的实现分离开了,但是OrderPromotion还是一直在变得,每一次添加或者下线活动都需要修改该类。

现在我们希望OrderPromotion是不变的,将PromotionCalculation的实例化剥离开来。创建类很明显是使用工厂设计模式了。

OrderPromotion

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
public class OrderPromotion {

    public BigDecimal promotion(Order order, int[] promotions){
        for(int promotion:promotions){
            PromotionFactory.getPromotionCalculate(promotion).calculate(order);
        }
        return order.getResultPrice();
    }
}

类的创建工作交给工厂来实现。

PromotionFactory

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
public class PromotionFactory {

    public static PromotionCalculate getPromotionCalculate(int promotion){
        switch (promotion){
            case 1:
                return new PromotionType1Calculate(order);
            break;
            case 2:
                return new PromotionType1Calculate(order);
            break;
            case 3:
                return new PromotionType1Calculate(order);
            break;
            //more promotion
        }
        return null;
    }
}

使用工厂模式后OrderPromotion类就不需要改了,每一次添加新的促销活动后只需要在工厂类中添加即可。

优化四:配置+反射

上面的代码还存在的问题在于每一次需要添加新的促销活动的时候还是需要修改工厂类中的代码,这里我们通过配置文件加反射的方式来解决。

定义映射配置文件

mapping.properties

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
1=design.order.PromotionType1Calculate
2=design.order.PromotionType2Calculate
3=design.order.PromotionType3Calculate

PromotionFactory

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
public class PromotionFactory {

    private static Map<Integer, String> mapping = new HashMap<Integer, String>();

    static {
        try {
            Properties pps = new Properties();
            pps.load(new FileInputStream("Test.properties"));
            Iterator<String> iterator = pps.stringPropertyNames().iterator();
            while(iterator.hasNext()){
                String key=iterator.next();
                mapping.put(Integer.valueOf(key), pps.getProperty(key));
            }
        } catch (IOException e) {
            e.printStackTrace();
        }
    }

    public static PromotionCalculate getPromotionCalculate(int promotion) throws Exception {
        if(mapping.containsKey(promotion)){
            String beanName = mapping.get(promotion);
            return Class.forName(beanName).newInstance();
        }
        return null;
    }
}

通过上面的代码就可以实现不改变已有代码的前提下实现对功能的灵活扩展。当然,这里的代码只是作为演示用的,实际上可以改进的地方还有不少,像最后反射效率较低,也可以通过其他的方式来实现。

小结

设计模式是我们一定要了解的东西,熟悉设计模式能让我们设计出易于扩展和维护的代码结构。但是并不是任何地方都需要上设计模式,应该结合我们的项目实际进行分析是否需要设计模式,使用哪种设计模式。

作者:宁愿呢

本文参与 腾讯云自媒体同步曝光计划,分享自微信公众号。
原始发表:2019-10-13,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 Java技术栈 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
23种设计模式(二)---策略设计模式
策略设计模式---透彻讲解 一. 什么是策略设计模式 设计模式有三种:创建型, 行为型, 结构型. 策略设计模式属于行为型. 为什么属于行为型呢? 来看看下面解释: 1.1 什么是策略呢? 什
用户7798898
2021/06/23
6200
23种设计模式(二)---策略设计模式
Python设计模式知多少
设计模式是前辈们经过相当长的一段时间的试验和错误总结出来的最佳实践。我找到的资料列举了以下这些设计模式:工厂模式、抽象工厂模式、单例模式、建造者模式、原型模式、适配器模式、桥接模式、过滤器模式、组合模式、装饰器模式、外观模式、享元模式、代理模式、责任链模式、命令模式、解释器模式、迭代器模式、中介者模式、备忘录模式、观察者模式、状态模式、空对象模式、策略模式、模板模式、访问者模式、MVC模式、业务代表模式、组合实体模式、数据访问对象模式、前端控制器模式、拦截过滤器模式、服务定位器模式、传输对象模式,共33种。
dongfanger
2021/05/24
5950
Python设计模式知多少
Java 抽象类与接口在 Java17 及以上版本中的现代应用实践解析
在Java 17及后续版本中,抽象类和接口的应用结合了记录类(Record)、密封类(Sealed Class)、模式匹配(Pattern Matching)等新特性,为代码设计带来了更多可能。下面通过实际案例展示这些技术的综合应用。
啦啦啦191
2025/06/22
700
Java 抽象类与接口在 Java17 及以上版本中的现代应用实践解析
大话设计模式笔记(二)——商品促销 策略模式
版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
逝兮诚
2019/10/30
1.2K0
大话设计模式笔记(二)——商品促销 策略模式
【经典案例】Python详解设计模式:策略模式
比如,超市做活动,如果你的购物积分满1000,就可以按兑换现金抵用券10元,如果购买同一商品满10件,就可以打9折,如果如果购买的金额超过500,就可以享受满减50元的优惠。这是三个不同的促销策略。
一墨编程学习
2019/05/10
9340
【经典案例】Python详解设计模式:策略模式
搞定设计模式1之策略模式
本例子下载地址:https://files.cnblogs.com/mengxin523/策略模式.rar
SAP梦心
2022/05/09
2650
装饰器模式实际运用_个人工作总结总结的格式
大家好,我是架构君,一个会写代码吟诗的架构师。今天说一说装饰器模式实际运用_个人工作总结总结的格式,希望能够帮助大家进步!!!
Java架构师必看
2022/08/15
4600
装饰器模式实际运用_个人工作总结总结的格式
大话设计模式--第二章 策略设计模式
现在有一个需求: 给商场做一个收银软件. 营业员根据客户购买的产品的单价和数量, 向客户打印小票。
用户7798898
2020/09/27
1.4K0
大话设计模式--第二章 策略设计模式
《我要进大厂》设计模式之一:策略模式
对于程序员来说,不管是跳槽不可避免的面试,还是想写出可读性、可维护性好的代码,设计模式都是必修课。
伍六七AI编程
2024/07/16
1510
《我要进大厂》设计模式之一:策略模式
Java设计模式-策略模式
策略模式属于对象的行为模式。其用意是针对一组算法,将每一个算法封装到具有共同接口的独立的类中,从而使得它们可以相互替换。策略模式使得算法可以在不影响到客户端的情况下发生变化。 --摘选自《JAVA与模式》
蒋老湿
2019/07/31
7840
【设计模式】策略模式 ( 简介 | 适用场景 | 优缺点 | 代码示例 )
策略模式 : 定义了 算法家族 , 分别 封装起来 , 让它们之间 , 可以 相互替换 , 此模式 让 算法的变化 不会影响到 使用算法的用户 ;
韩曙亮
2023/03/29
1.2K0
Python 中的设计模式详解之:策略模式
策略模式:定义一系列算法,把它们一一封装起来,并且使它们之间可以相互替换。此模式让算法的变化不会影响到使用算法的客户。
丹枫无迹
2019/05/05
7300
Python 中的设计模式详解之:策略模式
《大话设计模式》解读02-策略模式
本篇文章,来解读《大话设计模式》的第2章——策略模式。并通过Qt和C++代码实现实例代码的功能。
xxpcb
2024/06/16
1580
《大话设计模式》解读02-策略模式
Carson带你学设计模式:策略模式(Strategy Pattern)
定义一系列算法,将每个算法封装到具有公共接口的一系列策略类中,从而使它们可以相互替换,并让算法可以在不影响到客户端的情况下发生变化。
Carson.Ho
2022/03/25
5380
Carson带你学设计模式:策略模式(Strategy Pattern)
使用策略+工厂模式彻底干掉代码中的if else!
对于业务开发来说,业务逻辑的复杂是必然的,随着业务发展,需求只会越来越复杂,为了考虑到各种各样的情况,代码中不可避免的会出现很多if-else。
macrozheng
2019/10/31
4.6K2
设计模式学习之路一:策略模式
小弟最近在研究设计模式,准备边学边发博客,与众多大佬们交流学习,希望各位能够指出不足之处(废话不多说了,直接开花)
止术
2020/09/15
3180
漫话:如何给女朋友解释什么是策略模式?
周末无事,窝在家里面看《权力的游戏第八季》,看的很是津津有味,虽然感觉有一点点要烂尾,但是我还是忍不住要去看到底谁可以坐上铁王座。
Java识堂
2019/06/19
5060
漫话:如何给女朋友解释什么是策略模式?
设计模式之 - 策略落实
策略模式:它定义了算法家族,分别封装起来,让他们之间可以互相替换,此模式的变化,不会影响到使用算法的客户。
一个程序员的成长
2020/11/25
3790
设计模式之 - 策略落实
设计模式觉醒系列(01)设计模式的基石 | 六大原则的核心是什么?
随着工作年限的不断增长,在技术经验积累的路上,我们在技术框架、性能优化、业务系统架构研发、踩坑经验等方面上投入了非常多的时间。然而在具体的代码架构设计、代码复用性、可读性、可扩展性、可靠性上容易被忽略。而编码能力的底子,除了丰富研发实践经验,设计模式的融会贯通也同样重要。设计模式的思想就像参天大树的根基,对未来可以触达的高度有着举足轻重的影响。
拉丁解牛说技术
2025/02/24
2610
设计模式觉醒系列(01)设计模式的基石 | 六大原则的核心是什么?
策略模式(Strategy Pattern)- 最易懂的设计模式解析
定义一系列算法,将每个算法封装到具有公共接口的一系列策略类中,从而使它们可以相互替换,并让算法可以在不影响到客户端的情况下发生变化。
Carson.Ho
2019/02/22
7230
推荐阅读
相关推荐
23种设计模式(二)---策略设计模式
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
本文部分代码块支持一键运行,欢迎体验
本文部分代码块支持一键运行,欢迎体验