详解Java如何在业务代码中优雅的使用策略模式

 更新时间:2023年08月27日 10:40:48   作者:半亩方塘立身  
这篇文章主要为大家介绍了Java如何在业务代码中优雅的使用策略模式,文中的示例代码讲解详细,具有一定的学习价值,感兴趣的可以了解下

策略模式介绍

假设你正在开发一个电商平台,其中涉及到商品的折扣策略。优惠策略有很多种可能,如领取优惠券抵扣、返现促销、拼团优惠等。最初的实现可能会在购物车类中嵌入各种折扣逻辑,导致代码的可维护性和扩展性下降。

下面代码在业务开发中经常遇到,代码满足了业务需求,客户可根据自己的需求选择不同的优惠策略。但是,经过一段时间的业务积累,促销活动会越来越多。于是,程序员就开始经常加班,每次上活动之前都要通宵改代码,而且要做重复测试,判断逻辑可能也会变得越来越复杂。此时,我们要思考代码是否需要重构。

public static void main(String[] args) {
    PromotionActivity promotionActivity = null;
    String promotionKey = "COUPON";
    if(StringUtils.equals(promotionKey,"COUPON")){
        promotionActivity = new PromotionActivity(new CouponStrategy());
    }else if(StringUtils.equals(promotionKey,"CASHBACK")){
        promotionActivity = new PromotionActivity(new CashbackStrategy());
    }//......
    promotionActivity.execute();
}

下面我们看下策略模式如何优雅的解决这个问题。

策略模式是一种行为型设计模式,它允许在运行时选择算法的一种方式,使得算法可以独立于客户端代码进行变化。在业务代码中使用策略模式可以帮助你实现可维护、可扩展和可变化的代码结构。下面是在业务代码中使用策略模式的一般步骤:

  • 定义策略接口: 首先,定义一个策略接口,其中声明了策略的方法或行为。这些方法将在不同的具体策略类中实现。
  • 创建具体策略类: 创建实现策略接口的具体策略类,每个类实现了策略接口中的方法。每个具体策略类代表了一个算法或行为的具体实现。
  • 在客户端代码中使用策略: 在客户端代码中,通过持有策略接口类型的引用,可以在运行时选择不同的策略实现。这样客户端代码可以根据需要切换或替换不同的策略。

业务代码中如何使用

现在后端项目基本都是基于 Spring Boot 开发的,我们基于 Spring Boot 作为基础框架,教你如何使用 Spring 依赖注入的特性,优雅的实现策略模式。

既然是策略模式,那么定义策略肯定是首当其冲,策略我们使用枚举实现最佳

public enum StrategyType {
    STRATEGY_A(1, "策略A"),
    STRATEGY_B(2, "策略B");
    private int code;
    private String desc;
    StrategyType(int code, String desc) {
        this.code = code;
        this.desc = desc;
    }
    public int getCode() {
        return code;
    }
    public String getDesc() {
        return desc;
    }
}

再定义一个接口,接口有两个方法,getType用来获取策略的业务类型,execute用来执行业务

public interface Strategy {
    void execute();
    StrategyType getType();
}

这里我们实现个策略StrategyA

@Component("strategyA")
public class StrategyA implements Strategy {
    @Override
    public void execute() {
        System.out.println("Executing strategy A");
    }
    @Override
    public StrategyType getType() {
        return StrategyType.STRATEGY_A;
    }
}

再实现个策略StrategyB

@Component("strategyB")
public class StrategyB implements Strategy {
    @Override
    public void execute() {
        System.out.println("Executing strategy B");
    }
    @Override
    public StrategyType getType() {
        return StrategyType.STRATEGY_B;
    }
}

我们通过定义一个工厂类,然后使用 Spring 的依赖注入特性,可以注入一个接口的多个实现,这里采用 List<Strategy> 的形式注入,Spring 也支持通过 Map<String,Strategy> 的形式注入,如果使用 Map 注入,那么 key 就是类名,小伙伴们自己也可以测试一下~

为方便调用我们需要将List<Strategy>转换成Map<StrategyType, Strategy>结构,业务执行时可以直接传递业务参数(这里是我们定义的一个业务枚举StrategyType)获取Bean。这里我们直接使用Spring@PostConstruct注解在方法上,表示此方法是在Spring实例化该Bean之后马上执行此方法。

@Service
public class StrategyFactory {
    private Map<StrategyType, Strategy> strategyMap = new ConcurrentHashMap<>();
    @Resource
    private List<Strategy> strategyList;
    public void execute(StrategyType type) {
        strategyMap.get(type).execute();
    }
    @PostConstruct
    public void init() {
        for (Strategy strategy : strategyList) {
            strategyMap.put(strategy.getType(), strategy);
        }
    }
}

最后我们在业务类StrategyService直接使用就行了。

@Service
public class StrategyService {
    @Resource
    private StrategyFactory strategyFactory;
    public void executeStrategy(StrategyType type) {
        strategyFactory.execute(type);
    }
}

最终结构如下所示:

总结

使用 Spring 的依赖注入特性,可以注入一个接口的多个实现,很容易就实现了策略模式的选择,这样后续添加一种策略的时候,完全不需要改动主要逻辑,只需添加具体实现即可。

虽然我们是讲策略模式,其实里面还包含了工厂模式。

到此这篇关于详解Java如何在业务代码中优雅的使用策略模式的文章就介绍到这了,更多相关Java策略模式内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Intellij IDEA创建web项目的超详细步骤记录

    Intellij IDEA创建web项目的超详细步骤记录

    如果刚开始接触IDEA,或者之前使用的是eclipse/myEclipse的话,即使是创建一个JAVA WEB项目,估计也让很多人费了好几个小时,下面这篇文章主要给大家介绍了关于Intellij IDEA创建web项目的超详细步骤,需要的朋友可以参考下
    2022-08-08
  • javaweb判断当前请求是否为移动设备访问的方法

    javaweb判断当前请求是否为移动设备访问的方法

    这篇文章主要为大家详细介绍了javaweb判断当前请求是否为移动设备访问的方法,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-05-05
  • SpringBoot Controller返回图片的三种方式

    SpringBoot Controller返回图片的三种方式

    在互联网的世界里,图片无处不在,它们是信息传递的重要媒介,也是视觉盛宴的一部分,而在Spring Boot项目中,如何优雅地处理和返回图片数据,则成为了开发者们不得不面对的问题,今天,就让我们一起来探索Spring Boot Controller的神奇转换,需要的朋友可以参考下
    2024-07-07
  • JAVA线程池专题(概念和作用)

    JAVA线程池专题(概念和作用)

    这篇文章主要介绍了Java线程池的概念和作用,文中讲解非常详细,代码帮助大家更好的理解和学习,感兴趣的朋友可以了解下
    2020-06-06
  • idea实现类快捷生成接口方法示例

    idea实现类快捷生成接口方法示例

    这篇文章主要介绍了idea实现类快捷生成接口方法示例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-08-08
  • ActiveMQ整合Spring入门用法解析

    ActiveMQ整合Spring入门用法解析

    这篇文章主要介绍了ActiveMQ整合Spring入门用法解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2020-10-10
  • java System类和Arrays类详解

    java System类和Arrays类详解

    这篇文章主要介绍了java System类和Arrays类详解,文章围绕主题展开详细的内容介绍,具有一定的参考价值,需要的小伙伴可以参考一下
    2022-08-08
  • Java安全之Mojarra JSF反序列化讲解

    Java安全之Mojarra JSF反序列化讲解

    JSF 和类似的 Web 技术之间的区别在于 JSF 使用 ViewStates(除了会话)来存储视图的当前状态(例如,当前应该显示视图的哪些部分),这篇文章主要介绍了Java安全之Mojarra JSF反序列化知识讲解,包括漏洞复现和漏洞分析,需要的朋友可以参考下
    2022-11-11
  • Spring init-method与destroy-method属性的用法解析

    Spring init-method与destroy-method属性的用法解析

    这篇文章主要介绍了Spring init-method与destroy-method属性的用法,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-08-08
  • 分析Spring框架之设计与实现资源加载器

    分析Spring框架之设计与实现资源加载器

    Spring框架是由于软件开发的复杂性而创建的。然而,Spring的用途不仅仅限于服务器端的开发。从简单性、可测试性和松耦合性角度而言,绝大部分Java应用都可以从Spring中受益。今天来分析它的设计与实现资源加载器,从Spring.xml解析和注册Bean对象
    2021-06-06

最新评论