灵活应对:策略模式在软件设计中的应用
策略模式是一种行为型设计模式,它允许定义一系列算法,并将每个算法封装起来,使它们可以互换使用。策略模式让算法的变化独立于使用算法的客户端,使得在不修改原有代码的情况下切换或扩展新的算法成为可能。
使用策略模式的场景包括但不限于:
当存在多种实现方式,且需要在运行时根据不同条件动态选择具体实现时。例如,一个购物应用可能需要根据用户的会员等级来计算折扣,不同等级对应不同的计算方式,这时就可以使用策略模式来实现。
当存在一组类似的行为,实现细节略有不同,但又不希望通过继承来添加新的子类时。这样可以避免类的爆炸性增长,保持类的单一职责原则。
代码示例
在Java中实现策略模式时,通常会涉及以下几个角色的类和接口,它们之间的关系构成了策略模式的核心:
- 上下文(Context):
维护对策略对象的引用。可定义一个接口来让策略对象访问上下文中的其他数据。 - 策略Strategy):
定义所有支持的算法或行为的策略接口。该接口通常包含一个方法,该方法用于执行策略。 - 具体策略(Concrete Strategy):实现策略接口的具体类。每个具体策略类实现算法或行为的一个变体。
- 客户端(Client):
使用上下文和策略接口的类。不直接调用策略方法,而是通过上下文进行。
如下图所示:

以下是示例的java实现:
// 定义策略接口
interface Strategy {void execute();
}// 实现策略接口的具体策略类A
class ConcreteStrategyA implements Strategy {public void execute() {System.out.println("执行策略A");}
}// 实现策略接口的具体策略类B
class ConcreteStrategyB implements Strategy {public void execute() {System.out.println("执行策略B");}
}// 上下文类,用于维护策略对象
class Context {private Strategy strategy;public Context(Strategy strategy) {this.strategy = strategy;}public void setStrategy(Strategy strategy) {this.strategy = strategy;}public void executeStrategy() {strategy.execute();}
}// 客户端代码
public class Client {public static void main(String[] args) {// 创建具体策略对象A和BStrategy strategyA = new ConcreteStrategyA();Strategy strategyB = new ConcreteStrategyB();// 创建上下文对象,并设置具体策略对象Context context = new Context(strategyA);// 执行策略context.executeStrategy();// 切换策略context.setStrategy(strategyB);// 再次执行策略context.executeStrategy();}
}
JDK源码中策略模式的应用
Collections.sort() 方法在 JDK中是策略模式的一个经典应用。这个方法根据传入的列表(List)以及可选的比较器(Comparator),对列表进行排序。
使用流程:
- 客户端代码调用Collections.sort()方法,并传入自定义比较器。
- Collections.sort()方法接收列表作为参数。
- 列表返回其元素给Collections.sort()方法以进行比较。
- Collections.sort()方法使用自定义比较器(如果提供)或元素的自然顺序(如果元素实现了Comparable接口)来确定排序逻辑。
- 根据选择的策略,Collections.sort()方法对列表进行排序。
如图所示:

以下是 Collections.sort() 方法的策略模式分析:
-
策略接口:这里的策略接口是 Comparator,它定义了排序策略的公共行为,即 compare(Object o1, Object o2) 方法。Comparator 可以有多个实现,每个实现提供不同的排序规则。
-
具体策略:Comparator 的实现类代表具体策略。例如,Collections.reverseOrder() 返回一个反向排序的比较器,而 Collections.naturalOrder() 返回自然顺序的比较器。用户也可以自定义 Comparator 来表达特定的排序需求。
-
上下文(Context):Collections.sort() 方法本身充当上下文角色。它接受一个列表和一个可选的比较器对象。如果提供了比较器,Collections.sort() 会使用该比较器来对列表元素进行排序;如果没有提供,它会使用元素类型的自然顺序(如果元素类型实现了 Comparable 接口)。
-
策略的使用:在 Collections.sort() 内部,默认情况下,如果列表元素实现了 Comparable 接口,并且没有提供比较器,那么排序算法将使用 Comparable 接口提供的 compareTo() 方法作为排序策略。如果提供了 Comparator,则使用该比较器的 compare() 方法。
-
策略的切换:由于 Collections.sort() 能够接受不同的 Comparator 实现,因此可以在运行时动态地改变排序策略,无需修改排序代码本身。
-
算法的独立性:Collections.sort() 方法内部使用了归并排序或者TimSort(Java 7 引入),这个算法独立于策略。策略模式使得算法可以独立于具体的策略实现,增加了代码的灵活性和可扩展性。
优点
- 封装性:策略模式通过将算法封装在独立的策略类中,实现了算法与使用算法的客户端之间的解耦,提高了代码的模块性。
- 可扩展性:新的策略可以很容易地被添加进系统,符合开闭原则,即对扩展开放,对修改封闭。
- 动态替换:可以在运行时根据不同情况选择不同的算法策略,增加了系统的灵活性。
- 避免使用多重条件转移:策略模式提供了用组合来替代继承和庞大的条件语句的新思路,有利于代码的维护和理解。
缺点
- 系统复杂度:由于策略模式需要定义一系列的策略类,这会增加系统的复杂度。
- 数量增多引发的复杂性:随着策略数量的增加,客户端需要进行更多的策略选择和管理,这可能会引入额外的复杂性。
- 可能违反单一职责原则:如果策略类承担了过多的职责,或者某些策略实现过于复杂,可能会违背单一职责原则。
总的来说,策略模式有助于避免使用多重条件语句,随着策略类数量的增长,管理这些策略可能会变得复杂,且在某些情况下可能违反设计原则。因此,在应用策略模式时,应该权衡其带来的灵活性和解耦优势以及可能引入的复杂性和设计问题。
相关文章:
灵活应对:策略模式在软件设计中的应用
策略模式是一种行为型设计模式,它允许定义一系列算法,并将每个算法封装起来,使它们可以互换使用。策略模式让算法的变化独立于使用算法的客户端,使得在不修改原有代码的情况下切换或扩展新的算法成为可能。 使用策略模式的场景包…...
eosio.token 智能合约介绍
一、目的 eosio.token系统合约定义了允许用户为基于EOSIO的区块链创建、发行和管理代币的结构和操作,它演示了一种实现允许创建和管理代币的智能合约的方法。本文详细介绍了eosio.token系统合约并在本地测试链上实际发行了代币进行演示,适用于EOS智能合…...
3D 转换
1,3D的特点: 近小远大 物体后面遮挡不可见 2,3D移动 translate3d 3D移动在2D移动的基础上多加了一个可以移动的方向,就是z轴方向 transform:translateX(100px):仅仅是在x轴上移动…...
AI智能语音机器人安装方法
销售型的企业,基本靠电话营销拓客来实现效益的最大化。因为电销人员离职率高,且需求量大,需要接连不断的招人来实现业绩目标,电话机器人,贴牌招商,复制多个账户 这些都意味的企业的投入成本越来越大,博主从事多年AI技术…...
Python 潮流周刊#38:Django + Next.js 构建全栈项目
△△请给“Python猫”加星标 ,以免错过文章推送 你好,我是猫哥。这里每周分享优质的 Python、AI 及通用技术内容,大部分为英文。本周刊开源,欢迎投稿[1]。另有电报频道[2]作为副刊,补充发布更加丰富的资讯,…...
Jenkins升级后,构建任务配置界面重复错位
最近我把公司的Jenkins服务升级到了最新版本,升级完成后,点了一下构建任务,发现能够构建成功,就以为顺利完成升级了,下班走了,结果第二天,进入构建任务配置界面发现,界面一团乱麻&am…...
Python基础学习 -07 运算符
Python 运算符 运算符用于对变量和值执行操作。 Python 在以下组中划分运算符: 算术运算符赋值运算符比较运算符逻辑运算符身份运算符成员运算符位运算符 Python 算术运算符 算术运算符与数值一起使用来执行常见的数学运算: 运算符名称实例加x y-…...
Nim游戏
文章目录 题目描述输入格式输出格式 结论程序代码 题目描述 给定 n 堆石子,两位玩家轮流操作,每次操作可以从任意一堆石子中拿走任意数量的石子(可以拿完,但不能不拿),最后无法进行操作的人视为失败。 问…...
Pytorch: torch.linspace等间隔数值函数
torch.linspace 是 PyTorch 提供的一个用于生成等间隔数值的函数。具体而言,torch.linspace 会在指定的区间内生成指定数量的等间隔数值。 torch.linspace(start, end, steps100, dtypeNone, layouttorch.strided, deviceNone, requires_gradFalse)参数说明&#x…...
【C++】案例:数列求和 与 条件筛选
1.数列求和 题目: 设计一个程序,要求对数列2471116……n的前n项求和, 例如输入3,输出13; 输入6,输出62。 答案: #include <iostream>int main() {int n;std::cout << "请输入一个正…...
问题:下列哪些属于历史文化资源的特征( ). #学习方法#学习方法
问题:下列哪些属于历史文化资源的特征( ). A、稀缺性 B、脆弱性 C、可再生性 D、多样性 参考答案如图所示...
大数据 - Spark系列《四》- Spark分布式运行原理
Spark系列文章: 大数据 - Spark系列《一》- 从Hadoop到Spark:大数据计算引擎的演进-CSDN博客 大数据 - Spark系列《二》- 关于Spark在Idea中的一些常用配置-CSDN博客 大数据 - Spark系列《三》- 加载各种数据源创建RDD-CSDN博客 目录 🍠…...
Java使用规范
1.关键字 定义:被Java语言赋予了特殊含义,用做专门用途的字符串(单词) 特点:关键字中的所有字母都是小写 2.保留字 java保留字:现有Java版本尚未使用,但以后的版本可能会作为关键字使用。命名标识符时要避免使用这些…...
Debian 11 安装并开启SSH服务实现允许root用户使用SecureCRT远程登录
Debian11系统默认没有安装SSH服务,如需要开启远程登录则需要安装相应的服务。 确保你已经登录到Debian系统,并具有root用户或sudo特权。 打开终端,并使用以下命令安装OpenSSH服务器软件包: sudo apt update sudo apt install ope…...
Linux下对线程的理解(上)
1、线程的概念 要理解线程首先要理解页表和进程地址空间,我是这样子理解的,1、进程地址空间是进程访问资源的窗口。2、页表是规定进程地址空间中哪些属于进程。3、合理的使用进程地址空间页表可以对资源进行划分。而如何理解进程呢?进程是接受…...
【蓝桥杯】环形链表的约瑟夫问题
目录 题目描述: 输入描述: 输出描述: 示例1 解法一(C): 解法二(Cpp): 正文开始: 题目描述: 据说著名犹太历史学家 Josephus 有过以下故事&a…...
深度学习本科课程 实验1 Pytorch基本操作
一、Pytorch基本操作考察 1.1 任务内容 使用 𝐓𝐞𝐧𝐬𝐨𝐫 初始化一个 𝟏𝟑 的矩阵 𝑴 和一个 𝟐𝟏 的矩阵 𝑵,对两矩阵…...
大数据分析|设计大数据分析的三个阶段
文献来源:Saggi M K, Jain S. A survey towards an integration of big data analytics to big insights for value-creation[J]. Information Processing & Management, 2018, 54(5): 758-790. 下载链接:链接:https://pan.baidu.com/s/1…...
华为机考入门python3--(7)牛客7-取近似值
分类:数字 知识点: str转float float(str) 向上取整 math.ceil(float_num) 向下取整 math.floor(float_num) 题目来自【牛客】 import math def round_to_int(float_num): # 如果小数点后的数值大于等于0.5,则向上取整…...
C# Avalonia 11.0.6 绘图
在 Avalonia 11.0.6 中,Render 方法是被标记为 sealed 的,意味着不能直接在子类中重写这个方法。这样的设计可能是为了确保一致性和避免误用。 如果你需要在 Avalonia 中进行自定义的绘图操作,可以使用 DrawingContext,但是需要通…...
解决vue项目 vscode查找文件应用 ctrl+鼠标点击import无法跳转的问题
踩坑 前提是 AI的解决方案处理完,你的vue文件一体的script可以查看里面的import文件引用,但是独立的index.js-import无论如何都查看不了文件应用。 解决办法 如下是我的tscoonfig.json。 实际上就是加上 【“allowJs”: true, //为了查看文件引用&#x…...
一次 Spring 循环依赖源码走读:从三级缓存误用到 Bean 生命周期深度解析
在团队最近一次架构评审会上,关于 Spring 循环依赖的处理方式爆发了一场激烈争论。 “直接用 Lazy 不就行了?” 小李拍着桌子说,“我上个月在订单服务里就这么干的,上线一点问题没有。” “Lazy 只是绕开问题,不是解决…...
MAF快速入门(21)RC5引入的Script运行能力
大家好,我是Edison。最近我一直在跟着圣杰的《.NETAI智能体开发进阶》课程学习MAF开发多智能体工作流,我强烈推荐你也上车跟我一起出发!上一篇,我们了解下.NET 10新推出的File-Based App模式,它和MAF一起可以形成一个强…...
小白也能玩转Qwen3-TTS:一键部署多语言语音生成,实测效果惊艳
小白也能玩转Qwen3-TTS:一键部署多语言语音生成,实测效果惊艳 1. 为什么选择Qwen3-TTS 作为一个语音生成领域的从业者,我测试过市面上大多数TTS(文本转语音)模型,Qwen3-TTS-12Hz-1.7B-VoiceDesign给我留下…...
java+vue+SpringBoot企业信息管理系统(程序+数据库+报告+部署教程+答辩指导)
源代码数据库LW文档(1万字以上)开题报告答辩稿ppt部署教程代码讲解代码时间修改工具 技术实现 开发语言:后端:Java 前端:vue框架:springboot数据库:mysql 开发工具 JDK版本:JDK1.8 数…...
安全是跑出来的:从萝卜快跑看自动驾驶的“成人礼”
近日,武汉市区部分“萝卜快跑”自动驾驶车辆出现突发停驶异常状况,部分车辆在道路上停止运行,导致乘客被困、交通受阻。官方通报显示,此次事件为系统故障触发的车辆停滞,所有乘客已安全撤离,无人员伤亡。作…...
PowerToys中文版:三步搞定Windows效率工具的完全汉化体验
PowerToys中文版:三步搞定Windows效率工具的完全汉化体验 【免费下载链接】PowerToys-CN PowerToys Simplified Chinese Translation 微软增强工具箱 自制汉化 项目地址: https://gitcode.com/gh_mirrors/po/PowerToys-CN 你是否曾经因为PowerToys的英文界面…...
CosyVoice2-0.5B实战案例:跨境电商独立站商品页嵌入式语音播放功能实现
CosyVoice2-0.5B实战案例:跨境电商独立站商品页嵌入式语音播放功能实现 1. 引言:当商品介绍会“说话” 想象一下,你正在浏览一个海外独立站的商品页面,琳琅满目的图片和文字描述让你有些眼花缭乱。这时,你看到一个“…...
如何全面提升GTA5游戏体验:YimMenu安全使用与功能优化终极指南
如何全面提升GTA5游戏体验:YimMenu安全使用与功能优化终极指南 【免费下载链接】YimMenu YimMenu, a GTA V menu protecting against a wide ranges of the public crashes and improving the overall experience. 项目地址: https://gitcode.com/GitHub_Trending…...
OpenClaw安全防护指南:Qwen3-32B私有化部署下的权限管控策略
OpenClaw安全防护指南:Qwen3-32B私有化部署下的权限管控策略 1. 为什么需要关注OpenClaw的安全防护? 当我第一次把OpenClaw部署在自己的开发机上时,那种兴奋感至今记忆犹新——一个能帮我自动处理文件、整理资料、甚至写代码的AI助手&#…...
