一文简单全面了解策略模式的使用【花几分钟轻松掌握一个知识点】

简介: 您好,我是码农飞哥,感谢您阅读本文,欢迎一键三连哦。本文重点:介绍策略模式概念以及实际应用。干货满满,建议收藏,需要用到时常看看。小伙伴们如有问题及需要,欢迎踊跃留言哦~ ~ ~

您好,我是码农飞哥,感谢您阅读本文,欢迎一键三连哦

本文重点:介绍策略模式概念以及实际应用。

干货满满,建议收藏,需要用到时常看看。小伙伴们如有问题及需要,欢迎踊跃留言哦~ ~ ~

adf0919707d98576593188d4855455a7_watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3UwMTQ1MzQ4MDg=,size_16,color_FFFFFF,t_70.png

问题需求(支付系统渠道商的选择问题)

聚合支付项目对接了三个渠道商,分别渠道商A,渠道商B,渠道商C,每个渠道商都有一套独立的对接文档(从商户入驻,到支付,到清算)。那么该如何实现这个功能呢?

问题解决

V1.0版本

V1.0版本就是直接梭哈,将所有的业务逻辑写在了客户端,客户端在调用的时候可以分别调用各个渠道商的商户入驻方法。

public class Client {
    public static void main(String[] args) {
        Client client = new Client();
        client.addCustomerA("商户1");
        client.addCustomerB("商户2");
        client.addCustomerC("商户3");
    }
    public String addCustomerA(String customer) {
        System.out.println("渠道商A入驻"+customer+"成功");
        return "success";
    }
    public String addCustomerB(String customer) {
        System.out.println("渠道商B入驻"+customer+"成功");
        return "success";
    }
    public String addCustomerC(String customer) {
        System.out.println("渠道商C入驻"+customer+"成功");
        return "success";
    }
}

上述代码把策略和调用都放在了客户端中,从中不难看出,如果要在增加一个渠道商的话,则必定要修改客户端 增加新渠道商的商户入驻方法。这样就会导致客户端变得十分臃肿复杂维护起来非常麻烦,同时也没有体现面向对象的设计思想,在实际开发中写出这样的代码怕是要被老板优化。 那么我们该如何优化呢?

v1.1版本

从面向对象的角度出发,每个渠道商抽象成一个具体的对象,用一个单独的类保存其属性和行为,同时公共行为(商户入驻的行为)则用一个接口来定义,每个渠道商的类分别实现这个接口。

1.定义公共的接口Income,该接口主要是规范了各个渠道商的行为,这里通过addCustomer方法定义了商户入驻的行为。

public interface Income {
    /**
     * 商户入驻
     * @return
     */
    String addCustomer(String customer);
}

2.定义各个渠道商的类,这里分别定义了MerchantAIncome,MerchantBIncome以及MerchantCIncome 三个渠道商的类,并且这三个类都实现了Income接口的addCustomer方法,实现商户入驻。

public class MerchantAIncome implements Income {
    @Override
    public String addCustomer(String customer) {
        System.out.println("渠道商A入驻"+customer+"成功");
        return "success";
    }
}

3.定义环境类Context,这里的环境类主要的作用就是持有一个Income对象的引用,并且该Income对象所能操作的各种行为(即各种策略)。

public class Context {
    private Income income;
    public Context(Income income) {
        this.income = income;
    }
    public String addCustomer(String customer) {
        return income.addCustomer(customer);
    }
}

4.定义调用客户端

public class Client {
      public static void main(String[] args) {
        new Context(new MerchantAIncome()).addCustomer("商户1");
        new Context(new MerchantBIncome()).addCustomer("商户2");
        new Context(new MerchantCIncome()).addCustomer("商户3");
    }
}

UML图

上述代码的UML图如下图所示:这里有三个主要的角色:

1.环境类Context

2.抽象的策略Income接口

3.具体的策略MerchantAIncome,MerchantBIncome以及MerchantCIncome这三个类。

c4aebd834e3c2236e8fa54160df8f6cb_watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3UwMTQ1MzQ4MDg=,size_16,color_FFFFFF,t_70.png

如上图所示:每个渠道商的类都是实现了一个公共的接口,该接口定义了渠道商中各种行为,包括商户入驻等等。客户端判断当前使用的是哪个渠道商。然后,生成该渠道商的实例,接着调用该渠道商的相关方法。每个渠道商自身的变化不会影响到客户端。相比较于V1.0版本而言,代码逻辑简洁了很多,增加一个渠道商的话只需要再增加一个渠道商的类,并且在客户端中加上一个判断,符合开闭原则。这里就引出了本文将要介绍的模式----策略模式。

策略模式

定义

策略模式就是定义一系列的算法(对应本例中的一系列渠道商), 将每个算法封装到具有公共接口的一系列策略类中(对应到本例中就是将每个渠道商封装成一个单独的类,并且这些类实现一个公共的接口),从而使它们可以相互替换并且让算法可以在不影响客户端的情况下发生变化。

主要作用

从策略模式的定义中我们可以看出策略模式的主要作用就是将算法的责任和对象本身解耦,使得:

1.算法可以独立于客户端而变化(每个渠道商内部变化不影响客户端)

2.客户端可以根据外部条件而选择不同的策略来解决不同的问题。

优点

1.策略类之间可以自由切换

2.易于扩展,增加一个新的策略只需要添加一个具体的策略类即可,基本不需要改变原有的代码,符合“开闭原则”

3.消除了一些 if else条件语句

缺点

1.客户端必须知道所有的策略类,并自行决定使用哪个策略类。

2.策略模式将造成产生很多策略类,可以通过使用享元模式在一定程度上减少对象的数量。

策略模式与其它模式的比较

与状态模式的比较

策略模式的条件选择只执行一次,而状态模式是随着实例参数(对象实例的状态)的改变不停地更改执行模式。

与简单工厂模式的比较

工厂模式是创建型模式,它关注的是对象的创建,提供创建对象的接口,

策略模式是对象行为型模式,它关注的是行为和算法的封装。比如:出行方案中,策略模式是让你选择一种出行方式,而工厂模式是代替你构建具体的方案。

扩展延伸

v1.2版本

说完了策略模式,前面V1.1版本确实很好的运用到了策略模式,还是还不够,还是有点繁琐,客户端在调用的时候还是需要通过条件判断来决定使用哪种策略,这样还是没有干掉if和else,那么有没有什么方式可以干掉客户端的条件判断呢?答案是有的。我们可以通过枚举类来定义好各种渠道商的实例。

这里只需要新增一个枚举类

public enum MerchantIncomeEnum {
    MERCHANT_A("a", new MerchantAIncome()),
    MERCHANT_B("b", new MerchantBIncome()),
    MERCHANT_C("c", new MerchantCIncome()),;
    private String key;
    private Income income;
    MerchantIncomeEnum(String key, Income income) {
        this.key = key;
        this.income = income;
    }
    public static Income getIncome(String key) {
        Map<String, Income> map = Arrays.stream(MerchantIncomeEnum.values()).collect(Collectors.toMap(p -> p.key, p -> p.income));
        return map.get(key);
    }
}

修改环境类Context

public class Context {
    public String addCustomer(String key, String customer) {
        Income income = MerchantIncomeEnum.getIncome(key);
        return income.addCustomer(customer);
    }
}

这样客户端在调用的时候只需要传入指定渠道商的Key就可以调用指定渠道商的方法。

总结

本文由一个实际应用场景出发,引出了策略模式:策略模式主要解决的问题是如何将对象和算法分开,使得算法可以独立于使用它的客户端而变化。它的优点是易于扩展,策略类之间可以自由的切换。缺点是客户端在调用的时候必须要知道所有的策略类,并自行决定使用哪个策略类。同时策略模式会产生很多策略类。策略模式的适用于有一系列算法或者策略的场景下,比如:商场的各种折扣算法,出行的各种方案等等。


相关文章
|
JSON 前端开发 Java
几分钟带你快速了解SpringMVC框架理论知识!
几分钟带你快速了解SpringMVC框架理论知识!
|
存储 Java 数据库
几分钟带你快速了解Spring框架理论知识!
几分钟带你快速了解Spring框架理论知识!
|
5月前
|
Java 程序员 容器
十分钟搞懂依赖注入
依赖注入(DI)是软件开发中的关键技术,它将类的依赖关系转移至外部管理,提升了代码的模块化和可测试性。如同炒菜时调料不由厨师直接添加,而是由调料师准备并递送,程序员只需声明所需依赖,外部机制如DI容器会负责实例化并注入这些依赖。这样,类变得更灵活且易于测试,同时也促进了代码的模块化。例如,在Java中,可以通过构造函数注入Logger接口的不同实现(如ConsoleLogger和FileLogger),使Application类与其具体实现解耦。
|
8月前
|
安全 网络协议 网络安全
网络安全笔记整理,你花了多久弄明白架构设计
网络安全笔记整理,你花了多久弄明白架构设计
|
8月前
|
设计模式 Java 开发者
一目了然!谁能想到Java多线程设计模式竟然被图解,看完不服不行
多线程设计模式在Java编程中起着至关重要的作用,它能够有效提高程序的执行效率,使得程序在处理大量数据和复杂任务时更加高效。然而,对于初学者来说,理解和应用多线程设计模式可能是一项相当具有挑战性的任务。为了让读者更加轻松地掌握这一复杂主题,我们带着一种全新的图解方式,深入剖析Java多线程设计模式的精髓。
|
存储 编译器 Linux
千万不要错过的后端【纯干货】面试知识点整理 I I
千万不要错过的后端【纯干货】面试知识点整理 I I
101 0
|
存储 设计模式 编译器
千万不要错过的后端【纯干货】面试知识点整理 I
千万不要错过的后端【纯干货】面试知识点整理 I
102 0
|
前端开发 JavaScript
当下做前端开发,不算简单,这篇文章可以让少走很多弯路以及需要掌握的知识
当下做前端开发,不算简单,这篇文章可以让少走很多弯路以及需要掌握的知识
|
监控 算法 Java
花了好几个晚上整理的JVM知识点,吐血献出(一)(下)
花了好几个晚上整理的JVM知识点,吐血献出(一)(下)
161 0
花了好几个晚上整理的JVM知识点,吐血献出(一)(下)