什么是策略模式?为什么需要策略模式?

开发 后端
本文我们分析了策略模式,它通过将算法与使用算法的代码分离,提供了一种灵活且可扩展的方式来管理算法。

1. 策略模式是什么?

先让我们看看百科上的定义:

大概的意思是:

在计算机编程中,策略模式(strategy pattern)是一种行为软件设计模式,允许在运行时选择算法。代码不是直接实现单个算法,而是接收有关使用一系列算法中的哪个算法的运行时指令。

例如,对传入数据执行验证的类可能会使用策略模式来选择验证算法,具体取决于数据类型、数据源、用户选择或其他区分因素。这些因素在运行时之前是未知的,并且可能需要执行完全不同的验证。

2. 策略模式的角色

策略模式包含以下几个角色:

  • 上下文(Context):持有策略的引用,负责对外提供策略接口。
  • 策略接口(Strategy):定义了一个接口,用于所有具体策略的实现类。
  • 具体策略(ConcreteStrategy):实现策略接口的具体类,包含具体的算法实现。

策略模式的主要目标是将算法的实现与使用算法的代码进行分离,减少各部分之间的耦合度,可以在不影响上下文的情况下自由地修改或扩展算法。其模型图如下:

3. Java示例

上面将理论讲述了一通,对策略模式还是似懂非懂。为了更好地理解策略模式,下面我们通过一个简单的Java示例来演示策略模式的应用:

// 策略接口
interface PaymentStrategy {
    void pay(int amount);
}

// 具体策略类:信用卡支付
class CreditCardPayment implements PaymentStrategy {
    private String cardNumber;

    public CreditCardPayment(String cardNumber) {
        this.cardNumber = cardNumber;
    }

    @Override
    public void pay(int amount) {
        System.out.println("Paid " + amount + " using Credit Card: " + cardNumber);
    }
}

// 具体策略类:AliPal支付
class AliPalPayment implements PaymentStrategy {
    private String email;

    public AliPalPayment(String email) {
        this.email = email;
    }

    @Override
    public void pay(int amount) {
        System.out.println("Paid " + amount + " using AliPal: " + email);
    }
}

// 上下文
class ShoppingCart {
    private PaymentStrategy paymentStrategy;

    public void setPaymentStrategy(PaymentStrategy paymentStrategy) {
        this.paymentStrategy = paymentStrategy;
    }

    public void checkout(int amount) {
        if (paymentStrategy == null) {
            System.out.println("Payment strategy not set. Cannot proceed to checkout.");
            return;
        }
        paymentStrategy.pay(amount);
    }
}

// 测试策略模式
public class StrategyPatternDemo {
    public static void main(String[] args) {
        ShoppingCart cart = new ShoppingCart();

        // 使用信用卡支付
        cart.setPaymentStrategy(new CreditCardPayment("1234-5678-9101-1121"));
        cart.checkout(100);

        // 使用AliPal支付
        cart.setPaymentStrategy(new AliPalPayment("user@alipay.com"));
        cart.checkout(200);
    }
}

代码解析:

在这个示例中,我们有一个支付策略的接口 PaymentStrategy,它定义了 pay(int amount) 方法。我们还有两个具体策略类:CreditCardPayment 和 AliPalPayment,分别实现了不同的支付方式。

ShoppingCart 类是上下文类,用于设置不同的支付策略并进行结账。我们可以轻松地在运行时改变支付策略,而不需要修改 ShoppingCart 的实现。

看完上面的示例,是不是觉得策略模式和多态有点类似,对于一种抽象的方式(接口)可以按照不同的方式(子类)去实现。不过两者还是有差异:

  • 策略模式:策略模式是一种行为设计模式,它定义了一系列算法,并将每一个算法封装起来,使得它们可以互换。策略模式允许客户端在运行时选择具体的算法,而不需要修改相应的代码。
  • 多态:多态是指对象可以通过同一接口对不同类型的行为进行调用。多态通常通过基类的引用调用子类的方法来实现,可以在运行时动态地选择执行的对象类型。比如下面的示例图:

4.  优缺点

优点:

  • 开放/闭合原则:可以在不修改现有代码的情况下添加新策略。
  • 提高灵活性:可以在运行时选择和切换策略。
  • 减少代码重复:不同策略中相似的代码可以被上下文管理。

缺点:

  • 有很多策略类会增加系统复杂度:每增加一种策略都需要对应的类。
  • 客户端需要知道不同策略的具体实现才能选择:这可能导致需要在客户端中进行大量的条件判断。
  • 策略管理可能变得复杂:一旦有很多策略,如何选择和使用这些策略可能会增加复杂性。

5. 使用的框架

策略模式本身并不是 Java框架的一部分,但在一些流行的框架中可以看到其应用,例如:

  • Spring:Spring框架中可以使用策略模式来实现不同的策略,比如事务管理、缓存策略等。
  • Apache Commons Collections:提供了多种集合操作的策略实现。
  • Java标准库:例如在 java.util.Comparator 接口中,提供了排序时不同的比较策略。

6. 实际应用场景

策略模式广泛应用于许多场景中,一些典型的应用场景包括:

  • 支付处理:不同的支付方式(如信用卡、AliPal、网银等)。
  • 排序算法:动态选择不同的排序算法。
  • 折扣计算:根据不同的促销策略计算购物车中的折扣。
  • 文件处理:不同的文件压缩或文件处理策略。

7. 总结

本文,我们分析了策略模式,它通过将算法与使用算法的代码分离,提供了一种灵活且可扩展的方式来管理算法。尽管在某些情况下会增加系统的复杂度,但它的优点通常能够抵消这些缺点,使得这种模式在许多实际应用中非常有价值。对于想要设计可扩展性和灵活性的系统,策略模式是一个非常有效的设计方案。 

责任编辑:赵宁宁 来源: 猿java
相关推荐

2020-02-10 09:35:18

数据中心服务器技术

2011-07-22 09:31:11

2013-11-26 16:09:34

Android设计模式

2012-11-21 09:34:58

SaaS应用SaaS应用集成软件集成

2021-06-09 08:53:34

设计模式策略模式工厂模式

2015-09-08 13:39:10

JavaScript设计模式

2023-02-03 17:37:08

2011-07-20 14:04:42

.NET策略模式

2021-03-18 08:36:24

核心代码ACM模式面试ti

2011-06-14 13:31:00

FS模式

2022-02-13 00:13:26

云安全数据安全

2012-08-30 09:07:33

设计模式

2019-08-22 09:25:02

云计算云备份数据

2023-05-06 11:05:12

2024-10-06 12:56:36

Golang策略设计模式

2017-07-07 10:55:14

数据库MongoDB设计模式

2014-12-29 10:39:16

JS

2024-01-29 12:22:07

设计模式策略模式

2021-03-08 11:11:00

机器学习人工智能AI

2023-11-07 08:00:00

Kubernetes
点赞
收藏

51CTO技术栈公众号