Python 如何实现 Strategy 策略设计模式?什么是 Strategy 策略设计模式?

news2024/11/19 5:32:43

策略模式(Strategy Design Pattern)是一种对象行为型设计模式,它定义了一系列算法,并使得这些算法可以相互替换,使得客户端代码可以独立于算法的变化而变化。策略模式属于对象行为模式。

在这里插入图片描述

主要角色:

  1. 策略接口(Strategy): 定义了所有支持的算法的公共接口。通常由具体策略类实现。

  2. 具体策略类(ConcreteStrategy): 实现了策略接口,提供具体的算法实现。

  3. 环境类(Context): 维持一个对策略对象的引用,并在需要时调用策略对象的算法。通常客户端通过环境类来使用策略。

工作流程:

  1. 客户端创建一个具体的策略对象,实现了策略接口。

  2. 客户端将策略对象传递给环境类。

  3. 环境类在需要执行算法的时候,调用策略对象的算法。

优点:

  • 算法可自由切换: 客户端可以根据需要切换不同的策略,灵活地选择合适的算法。

  • 避免使用多重条件语句: 策略模式可以避免使用过多的条件语句,提高代码的可维护性。

  • 符合开闭原则: 新的算法可以方便地加入到系统中,而不影响现有代码。

Python 示例代码(一):

以下是一个简单的策略模式的示例,假设有一个支付系统,可以选择不同的支付策略:

from abc import ABC, abstractmethod

# 策略接口
class PaymentStrategy(ABC):
    @abstractmethod
    def pay(self, amount):
        pass

# 具体策略类
class CreditCardPayment(PaymentStrategy):
    def pay(self, amount):
        print(f"Paying {amount} using Credit Card")

class PayPalPayment(PaymentStrategy):
    def pay(self, amount):
        print(f"Paying {amount} using PayPal")

# 环境类
class ShoppingCart:
    def __init__(self, payment_strategy):
        self.payment_strategy = payment_strategy
    
    def checkout(self, amount):
        self.payment_strategy.pay(amount)

# 客户端
credit_card_payment = CreditCardPayment()
paypal_payment = PayPalPayment()

cart1 = ShoppingCart(credit_card_payment)
cart1.checkout(100)

cart2 = ShoppingCart(paypal_payment)
cart2.checkout(150)

在这个例子中,PaymentStrategy 是策略接口,CreditCardPaymentPayPalPayment 是具体的策略类,而 ShoppingCart 是环境类。客户端可以根据需要选择不同的支付策略。


Python 示例代码(二)

让我们考虑一个动物王国的场景,其中有不同种类的动物,每种动物都有自己的叫声。我们可以使用策略模式来模拟这个场景:

from abc import ABC, abstractmethod

# 策略接口
class SoundStrategy(ABC):
    @abstractmethod
    def make_sound(self):
        pass

# 具体策略类
class BarkStrategy(SoundStrategy):
    def make_sound(self):
        return "Woof! Woof!"

class MeowStrategy(SoundStrategy):
    def make_sound(self):
        return "Meow!"

class RoarStrategy(SoundStrategy):
    def make_sound(self):
        return "Roar!"

# 环境类
class Animal:
    def __init__(self, sound_strategy):
        self.sound_strategy = sound_strategy
    
    def make_sound(self):
        return self.sound_strategy.make_sound()

# 客户端
dog = Animal(BarkStrategy())
cat = Animal(MeowStrategy())
lion = Animal(RoarStrategy())

print("Dog says:", dog.make_sound())
print("Cat says:", cat.make_sound())
print("Lion says:", lion.make_sound())

在这个例子中,SoundStrategy 是策略接口,定义了动物叫声的标准。BarkStrategyMeowStrategyRoarStrategy 是具体的策略类,分别模拟了狗、猫和狮子的叫声。Animal 是环境类,它接受一个叫声策略,并在调用 make_sound 时调用相应的策略。

在客户端,你可以为不同的动物选择不同的叫声策略,从而模拟它们各自的叫声行为。

这个例子更加生动,展示了策略模式在真实场景中的一种应用。这种方式使得系统更容易扩展,例如当新的动物种类被引入时,只需添加一个新的策略类即可。


使用该模式应该注意的地方:

使用策略模式时,需要注意一些关键点,以确保模式的有效实施和系统的可维护性。以下是一些建议:

  1. 接口设计: 确保策略接口(抽象策略类)清晰简洁,包含了所有具体策略类应实现的方法。这有助于确保一致性和易于理解。

  2. 灵活性与复杂性的平衡: 策略模式使得算法可以独立变化,但也可能导致类的数目增加。在设计时要平衡灵活性和类的数量,避免过度复杂化。

  3. 运行时切换策略: 考虑系统是否需要在运行时动态切换策略。如果需要,确保环境类(Context)能够接受新的策略并进行切换。

  4. 策略参数化: 确保策略能够接受参数,使得策略的行为可以根据输入参数进行调整。这提高了策略的灵活性。

  5. 合适的使用场景: 策略模式适用于一组算法变体,客户端需要从中选择一个的情况。如果只有一个算法,或者算法数量有限且变动不大,可能不是最适合的模式。

  6. 避免过度使用: 不要在不需要的地方引入策略模式。只有当有多个相关的算法时,而且需要在运行时选择时,才考虑使用策略模式。

  7. 关注性能: 策略模式可能引入一些额外的开销,因为需要通过接口调用来执行算法。在性能敏感的应用中,需要进行评估并选择合适的设计方案。

  8. 适当的文档: 提供适当的文档和注释,以解释每个具体策略的用途和行为。这有助于其他开发人员理解和使用你的代码。

  9. 测试: 对策略模式进行适当的单元测试,确保每个具体策略和整个系统的行为都符合预期。

  10. 考虑与其他模式结合使用: 策略模式可以与其他设计模式结合使用,例如工厂模式、单例模式等,以更好地满足系统需求。

在设计和使用策略模式时,综合考虑这些因素将有助于确保系统的可维护性、灵活性和性能。


本文就到这里了,感谢您的阅读 。别忘了点赞、收藏~ Thanks♪(・ω・)ノ 🍇

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1204601.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

postman调用接口报{“detail“:“Method \“DELETE\“ not allowed.“}错误, 解决记录

项目是python代码开发, urls.py 路由中访问路径代码如下: urlpatterns [path(reportmanagement/<int:pk>/, views.ReportManagementDetail.as_view(), namereport-management-detail),] 对应view视图中代码如下: class ReportManagementDetail(GenericAPIView):"…

cgo与调用c的回调函数指针

cgo直接调用函数&#xff0c;使用基本数据类型非常简单&#xff0c;包括一些结构体也比较简单&#xff0c;嵌套的稍微复杂些&#xff0c;但也可以&#xff0c;但有的时候&#xff0c;cgo调用c函数&#xff0c;会需要传递一个回调函数的指针&#xff0c;这时候就比较复杂了&…

Arcgis打开报错error code=-15

Provide your license server administrator with the following information: Error Code -15 问题描述 原因 长时间闲置后&#xff0c;license server administrator会关闭服务。再次打开之后会出现这个报错 解决方案 重启或者按下述做法&#xff1a; 打开任务管理器&am…

【无标题】通用工作站设计方案:ORI-D3R600服务器-多路PCIe3.0的双CPU通用工作站

ORI-D3R600服务器-多路PCIe3.0的双CPU通用工作站 一、机箱功能和技术指标&#xff1a; 系统 系统型号 ORI-SR630 主板支持 EEB(12*13)/CEB(12*10.5)/ATX(12*9.6)/Micro ATX 前置硬盘 最大支持8个3.5寸(兼容25寸)SATA硬盘 2*2.5(后置) 电源类型 CRPS元余电源&#xff0…

【Verilog语法】

Verilog语法 1. Verilog语法1.1 拼接运算符1.2 运算符优先级1.3 注释1.4 关键字1.5 模块结构1.6 结构语句1.7 赋值语句1.8 条件语句1.9 状态机1.10 OSI七层模型 1. Verilog语法 1.1 拼接运算符 1.2 运算符优先级 1.3 注释 1.4 关键字 1.5 模块结构 1.6 结构语句 1.7 赋值语句 …

【Java 进阶篇】JQuery DOM操作:Class属性的舞蹈魔法

在前端的世界中&#xff0c;JQuery如同一位舞者&#xff0c;通过灵活的舞步为我们展示了操纵HTML元素的艺术。而在这场舞蹈的精彩演出中&#xff0c;Class属性的操作是一项极富魅力的技艺。在本篇博客中&#xff0c;我们将深入研究JQuery DOM操作中的Class属性操作&#xff0c;…

pytest-bdd快速示例和问题解决

BDD 与 pytest-bdd BDD 即 Behavior-driven development&#xff0c;行为驱动开发。BDD行为驱动是一种敏捷开发模式, 重点在于消除开发/测试对需求了解的歧义及用户场景的验证。 pytest-bdd 是一个BDD测试框架&#xff0c;类似于behave, cucumber。它可以统一单元测试和功能测…

【数据结构】——单链表(增删查改)

目录 前言&#xff1a; 一&#xff1a;单链表的特点 ​编辑 二&#xff1a;单链表实现 单链表定义 2.1申请节点&#xff08;初始化&#xff09; 2.2单链表尾插 ​编辑 2.3单链表打印 2.4单链表头插 2.5单链表尾删 2.6单链表头删 2.7单链表查找 2.8在目标位置后面插入…

三国杀中的概率学问题4——曹冲

前言 这篇文章是围绕曹冲的称象技能展开的一些数学上的讨论&#xff0c;将涉及到积分、概率论等知识&#xff0c;并会做很多拓展。 值得说明的是&#xff0c;本文受到了这篇文章的一些启发。 连续情形1 先来看一个连续情形的问题。 问题一&#xff1a;假设每张牌的点数是0~1…

JVM:卡表元素如何维护?(写屏障)

写屏障 上面使用记忆集解决了缩减GC Roots扫描范围的问题&#xff0c;现在又抛出来一个新的问题&#xff0c;卡表元素如何维护的呢&#xff1f;&#xff0c;例如它们何时变脏、谁来把它们变脏等。 何时变脏这个问题应该很明确的&#xff0c;原则上应该发生在引用类型字段赋值…

编译原理-语法分析-自上而下分析

文章目录 语法分析器的功能自上而下分析面临的问题LL&#xff08;1&#xff09;分析法左递归的消除直接左递归非直接左递归 消除左递归的算法消除回溯、提左因子FIRST提左因子FOLLOW集 LL(1)的分析条件LL(1)文法构造FIRST和FOLLOW集合构造每个文法符号的FIRST集合构造FOLLOW集合…

java网络编程之UDP协议

文章目录 UDP简介一发一收客户端&#xff1a;服务端&#xff1a; 多发多收实现多开客户端&#xff1a;服务端 UDP简介 UDP&#xff08;User Datagram Protocol&#xff09; DatagramSocket 用于创建客户端、服务端DatagramSocket() :创建客户端的Socket对象&#xff0c;系统随…

C++ 11 新特性

目录 1. 支持特性的编译器版本2. 模板表达式中空格3. 空指针4. auto5. 统一初始化6. explict7. 范围for8. default&#xff0c;delete9. 化名模板&#xff08;alias template&#xff09;10. using11. noexcept12. override13. final14. decltype15. lambda16. Variadic Templa…

4.2每日一题(求多元函数在某一点的微分)

1、分别求x和y的偏导&#xff0c;再相加即可 2、因为多元函数的表达式不方便求偏导&#xff0c;所以可以使用先代后求法&#xff1a; &#xff08;1&#xff09;对x偏导&#xff1a;把y0代入&#xff0c;很容易求出对x偏导的结果 &#xff08;2&#xff09;对y偏导&#xff1a…

Django——路由层

一. 路由匹配 1. 路由匹配注意事项 urlpatterns [url(r^admin/, admin.site.urls),# 首页url(r^$,views.home),# 路由匹配url(r^test/$,views.test),url(r^testadd/$,views.testadd),# 尾页(了解): 后期使用异常捕获处理, 这样的尾页让django的第二次在路径中斜杠APPEND_SL…

Redis解决缓存问题

目录 一、引言二、缓存三、Redis缓存四、缓存一致性1.缓存更新策略2.主动更新 五、缓存穿透六、缓存雪崩七、缓存击穿1.基于互斥锁解决具体业务2.基于逻辑过期解决具体业务 一、引言 在一些大型的网站中会有十分庞大的用户访问流量&#xff0c;而过多的用户访问对我们的MySQL数…

Linux SSH免密登录

目录 简介 创建Linux用户和用户组 配置LINUX静态IP 编辑IP映射 SSH免密登录配置 登录测试 简介 SSH&#xff08;Secure shell&#xff09;是可以在应用程序中提供安全通信的一个协议&#xff0c;通过SSH可以安全地进行网络数据传输&#xff0c;它的主要原理是利用非对称加密…

Libhybris之线程局部存储TLS实例(五)

简介&#xff1a; CSDN博客专家&#xff0c;专注Android/Linux系统&#xff0c;分享多mic语音方案、音视频、编解码等技术&#xff0c;与大家一起成长&#xff01; 优质专栏&#xff1a;Audio工程师进阶系列【原创干货持续更新中……】&#x1f680; 人生格言&#xff1a; 人生…

二十三、W5100S/W5500+RP2040树莓派Pico<Web I/O 通过网页控制板载LED灯>

文章目录 1 前言2 简介2 .1 什么是Web&#xff1f;2.2 Web的优点2.3 Web数据交互原理2.4 Web应用场景 3 WIZnet以太网芯片4 HTTP网络设置示例概述以及使用4.1 流程图4.2 准备工作核心4.3 连接方式4.4 主要代码概述4.5 结果演示 5 注意事项6 相关链接 1 前言 Web只是一个静态的文…