设计模式- 策略模式(Strategy Pattern)结构|原理|优缺点|场景|示例

本文主要是介绍设计模式- 策略模式(Strategy Pattern)结构|原理|优缺点|场景|示例,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

                                     设计模式(分类)        设计模式(六大原则)   

    创建型(5种)        工厂方法         抽象工厂模式        单例模式        建造者模式        原型模式

    结构型(7种)        适配器模式        装饰器模式        代理模式        ​​​​​​外观模式      桥接模式        组合模式       享元模式

    行为型(11种)      策略模式        模板方法模式        观察者模式        迭代器模式        责任链模式        命令模式

                                   备忘录模式          状态模式          访问者模式        中介者模式


设计模式中的策略模式(Strategy Pattern)是一种行为设计模式,它定义了一系列算法,并将每个算法封装在一个单独的类中,使得它们可以互相替换。策略模式使得算法可以在运行时根据需要动态地改变,同时客户端代码可以通过统一的接口调用不同的策略实现。

模式结构

  1. 策略接口(Strategy Interface)

    • 定义所有支持的策略或算法所共有的方法签名,这是所有具体策略类的抽象父类或接口。
  2. 具体策略类(Concrete Strategies)

    • 每个具体策略类实现了策略接口,并提供了算法的具体实现。
    • 在具体策略类中包含了算法的详细逻辑。
  3. 上下文(Context)

    • 上下文是使用策略的对象,它维持对策略对象的引用,并定义了如何使用策略的方法。
    • 上下文可以根据需求改变策略,通常是通过策略接口设置具体的策略对象。

工作原理

  • 客户端:创建并配置上下文对象,指定要使用的具体策略。
  • 上下文:根据客户端的配置,保存一个指向具体策略对象的引用,并在需要执行策略时调用策略接口定义的方法。
  • 具体策略:执行实际的算法或行为。

优缺点

优点
  • 开放封闭原则:策略模式允许在不修改现有代码的基础上新增策略。
  • 多态性:客户端通过策略接口调用方法,无需关注具体实现细节,增强了代码的灵活性和可扩展性。
  • 解耦:策略模式将算法从使用它的上下文中解耦出来,便于算法的独立管理和测试。
缺点
  • 策略类数量增多:随着策略数量的增加,可能会产生大量的策略类。
  • 上下文需了解策略:虽然上下文不用关心策略的具体实现,但是它至少需要知道有哪些策略可供选择,并能够适配不同的策略。

适用场景

  • 系统需要多种算法解决同一问题,且在运行时可以动态切换算法
  • 算法的实现可以相互独立,互不影响
  • 希望避免使用多重条件判断(例如 switch-case 或 if-else)来选择算法

代码示例(以Java为例)

// 抽象策略接口
public interface DiscountStrategy {double getDiscount(double price);
}// 具体策略类 - 无折扣策略
public class NoDiscountStrategy implements DiscountStrategy {@Overridepublic double getDiscount(double price) {return price;}
}// 具体策略类 - 普通会员折扣策略
public class NormalMemberDiscountStrategy implements DiscountStrategy {@Overridepublic double getDiscount(double price) {return price * 0.95; // 九五折}
}// 具体策略类 - VIP会员折扣策略
public class VIPDiscountStrategy implements DiscountStrategy {@Overridepublic double getDiscount(double price) {return price * 0.85; // 八五折}
}// 上下文类 - 订单类,使用策略来计算折扣后的价格
public class Order {private DiscountStrategy discountStrategy;public Order(DiscountStrategy discountStrategy) {this.discountStrategy = discountStrategy;}public void setDiscountStrategy(DiscountStrategy discountStrategy) {this.discountStrategy = discountStrategy;}public double calculateFinalPrice(double originalPrice) {return discountStrategy.getDiscount(originalPrice);}
}// 客户端代码
public class Client {public static void main(String[] args) {Order orderWithoutDiscount = new Order(new NoDiscountStrategy());System.out.println("No discount applied: " + orderWithoutDiscount.calculateFinalPrice(100));Order normalOrder = new Order(new NormalMemberDiscountStrategy());System.out.println("Normal member discount applied: " + normalOrder.calculateFinalPrice(100));Order vipOrder = new Order(new VIPDiscountStrategy());System.out.println("VIP member discount applied: " + vipOrder.calculateFinalPrice(100));}
}

代码示例(以Python为例)

# 策略接口
from abc import ABC, abstractmethodclass PaymentStrategy(ABC):@abstractmethoddef pay(self, amount: float) -> None:pass# 具体策略类
class CreditCardPayment(PaymentStrategy):def __init__(self, card_number: str, cvv: str):self.card_number = card_numberself.cvv = cvvdef pay(self, amount: float) -> None:print(f"Paid {amount} using credit card ({self.card_number})")class PayPalPayment(PaymentStrategy):def __init__(self, account_id: str):self.account_id = account_iddef pay(self, amount: float) -> None:print(f"Paid {amount} using PayPal account ({self.account_id})")# 上下文
class ShoppingCart:def __init__(self, payment_strategy: PaymentStrategy):self.payment_strategy = payment_strategydef set_payment_strategy(self, strategy: PaymentStrategy) -> None:self.payment_strategy = strategydef checkout(self, total_amount: float) -> None:print(f"Checking out with total amount: {total_amount}")self.payment_strategy.pay(total_amount)# 客户端代码
cart = ShoppingCart(CreditCardPayment("1234567890123456", "123"))
cart.checkout(100.00)cart.set_payment_strategy(PayPalPayment("buyer@example.com"))
cart.checkout(200.00)

在这个示例中:

  • PaymentStrategy是策略接口,定义了支付方法pay
  • CreditCardPaymentPayPalPayment是具体策略类,实现了支付方式。
  • ShoppingCart是上下文,持有一个支付策略对象,并在其checkout方法中调用策略对象的pay方法来完成支付。根据需要,可以随时更改支付策略。

这篇关于设计模式- 策略模式(Strategy Pattern)结构|原理|优缺点|场景|示例的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/932189

相关文章

SpringBoot中SM2公钥加密、私钥解密的实现示例详解

《SpringBoot中SM2公钥加密、私钥解密的实现示例详解》本文介绍了如何在SpringBoot项目中实现SM2公钥加密和私钥解密的功能,通过使用Hutool库和BouncyCastle依赖,简化... 目录一、前言1、加密信息(示例)2、加密结果(示例)二、实现代码1、yml文件配置2、创建SM2工具

MySQL 定时新增分区的实现示例

《MySQL定时新增分区的实现示例》本文主要介绍了通过存储过程和定时任务实现MySQL分区的自动创建,解决大数据量下手动维护的繁琐问题,具有一定的参考价值,感兴趣的可以了解一下... mysql创建好分区之后,有时候会需要自动创建分区。比如,一些表数据量非常大,有些数据是热点数据,按照日期分区MululbU

MyBatis-Plus 中 nested() 与 and() 方法详解(最佳实践场景)

《MyBatis-Plus中nested()与and()方法详解(最佳实践场景)》在MyBatis-Plus的条件构造器中,nested()和and()都是用于构建复杂查询条件的关键方法,但... 目录MyBATis-Plus 中nested()与and()方法详解一、核心区别对比二、方法详解1.and()

Python函数作用域示例详解

《Python函数作用域示例详解》本文介绍了Python中的LEGB作用域规则,详细解析了变量查找的四个层级,通过具体代码示例,展示了各层级的变量访问规则和特性,对python函数作用域相关知识感兴趣... 目录一、LEGB 规则二、作用域实例2.1 局部作用域(Local)2.2 闭包作用域(Enclos

Java设计模式---迭代器模式(Iterator)解读

《Java设计模式---迭代器模式(Iterator)解读》:本文主要介绍Java设计模式---迭代器模式(Iterator),具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,... 目录1、迭代器(Iterator)1.1、结构1.2、常用方法1.3、本质1、解耦集合与遍历逻辑2、统一

Java 线程安全与 volatile与单例模式问题及解决方案

《Java线程安全与volatile与单例模式问题及解决方案》文章主要讲解线程安全问题的五个成因(调度随机、变量修改、非原子操作、内存可见性、指令重排序)及解决方案,强调使用volatile关键字... 目录什么是线程安全线程安全问题的产生与解决方案线程的调度是随机的多个线程对同一个变量进行修改线程的修改操

从原理到实战深入理解Java 断言assert

《从原理到实战深入理解Java断言assert》本文深入解析Java断言机制,涵盖语法、工作原理、启用方式及与异常的区别,推荐用于开发阶段的条件检查与状态验证,并强调生产环境应使用参数验证工具类替代... 目录深入理解 Java 断言(assert):从原理到实战引言:为什么需要断言?一、断言基础1.1 语

C++20管道运算符的实现示例

《C++20管道运算符的实现示例》本文简要介绍C++20管道运算符的使用与实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录标准库的管道运算符使用自己实现类似的管道运算符我们不打算介绍太多,因为它实际属于c++20最为重要的

Java中调用数据库存储过程的示例代码

《Java中调用数据库存储过程的示例代码》本文介绍Java通过JDBC调用数据库存储过程的方法,涵盖参数类型、执行步骤及数据库差异,需注意异常处理与资源管理,以优化性能并实现复杂业务逻辑,感兴趣的朋友... 目录一、存储过程概述二、Java调用存储过程的基本javascript步骤三、Java调用存储过程示

ModelMapper基本使用和常见场景示例详解

《ModelMapper基本使用和常见场景示例详解》ModelMapper是Java对象映射库,支持自动映射、自定义规则、集合转换及高级配置(如匹配策略、转换器),可集成SpringBoot,减少样板... 目录1. 添加依赖2. 基本用法示例:简单对象映射3. 自定义映射规则4. 集合映射5. 高级配置匹