【设计模式】三十二、策略模式

系列文章|源码

https://github.com/tyronczt/design-mode-learn

文章目录

    • 系列文章|源码
    • 一、模式定义与核心思想
    • 二、模式结构与Java实现
      • 1. 核心角色
      • 2. Java代码示例
    • 三、策略模式的五大核心优势
    • 四、适用场景
    • 五、与其他模式的对比
    • 六、最佳实践建议
    • 总结
  • 🚀进阶版【更符合实际开发】🚀
    • 策略+工厂模式
      • 效果图
      • 类图
      • 控制器
      • 策略工厂
      • 策略接口
      • 策略实现类
      • 枚举类
      • Spring工具类

一、模式定义与核心思想

策略模式(Strategy Pattern) 是一种行为型设计模式,其核心思想是将一组算法或行为封装为独立的策略类,使得它们可以相互替换,且算法的变化独立于使用它的客户端。该模式通过组合替代继承,符合开闭原则(对扩展开放、对修改关闭)。

二、模式结构与Java实现

1. 核心角色

策略接口(Strategy):定义算法族的公共行为(如PaymentStrategy接口)。
具体策略类(Concrete Strategy):实现接口的具体算法(如微信支付、支付宝支付)。
上下文类(Context):持有策略引用,动态切换算法(如PaymentContext类)。

2. Java代码示例

以下以支付方式选择为例展示实现:

// 1. 策略接口(定义支付行为)
public interface IPaymentStrategy {void pay(double amount);
}// 2. 具体策略类(微信支付)
public class WeChatPayStrategy implements IPaymentStrategy {@Overridepublic void pay(double amount) {System.out.println("微信支付:" + amount + "元");}
}// 3. 具体策略类(支付宝支付)
public class AliPayStrategy implements IPaymentStrategy {@Overridepublic void pay(double amount) {System.out.println("支付宝支付:" + amount + "元");}
}// 4. 上下文类(管理策略)
public class PaymentContext {private IPaymentStrategy strategy;public void setStrategy(IPaymentStrategy strategy) {this.strategy = strategy;}public void executePayment(double amount) {if (strategy != null) {strategy.pay(amount);} else {throw new IllegalStateException("未选择支付方式");}}
}// 5. 客户端调用
public class Client {public static void main(String[] args) {PaymentContext context = new PaymentContext();context.setStrategy(new WeChatPayStrategy());context.executePayment(100.0);  // 输出:微信支付:100.0元context.setStrategy(new AliPayStrategy());context.executePayment(200.0);  // 输出:支付宝支付:200.0元}
}

三、策略模式的五大核心优势

  1. 消除复杂条件分支
    • 传统方式需通过大量if-else判断不同场景(如支付方式选择),策略模式将每个算法封装为独立类,代码更清晰。
  2. 符合开闭原则(OCP)
    扩展性高:新增策略(如银行卡支付)只需添加新类,无需修改现有代码。
    维护性高:修改某一策略的逻辑不会影响其他策略。
  3. 动态切换算法
    • 运行时通过setStrategy()方法灵活切换策略(如根据用户选择切换支付方式)。
  4. 算法复用与解耦
    • 不同上下文可共享同一策略(如多个订单系统共用支付策略)。
    • 算法实现与业务逻辑解耦,便于单元测试。
  5. 减少重复代码
    • 公共逻辑可抽取至抽象类(如支付前的参数校验),具体策略类仅关注核心算法。

四、适用场景

  1. 多算法动态选择:如支付方式、数据加密算法、排序算法等。
  2. 业务规则频繁变化:如电商促销策略(满减、折扣、积分)。
  3. 替代复杂条件分支:避免多个if-elseswitch-case语句。

五、与其他模式的对比

模式核心区别典型场景
策略模式关注算法族动态切换支付方式、折扣策略
状态模式关注状态驱动的行为变化订单状态流转、游戏角色
工厂模式关注对象创建策略类的实例化管理

六、最佳实践建议

结合工厂模式:使用策略工厂统一管理策略实例(如通过Map存储策略类)。
默认策略:在上下文中设置默认策略以避免空指针异常。
策略组合:复杂场景可将多个策略组合使用(如先满减再折扣)。

总结

策略模式通过封装算法动态切换机制,显著提升了代码的可维护性与扩展性。在Java中,结合接口与多态特性,能高效实现复杂的业务需求(如电商支付、促销活动)。其核心价值在于将“做什么”与“怎么做”分离,是应对算法灵活性和业务变化的利器。

🚀进阶版【更符合实际开发】🚀

策略+工厂模式

效果图

实际开发中只要前端传入支付类型,后端即可根据策略进行相应支付逻辑处理。

类图

控制器

@RestController
public class AdvancedClientController {@Resourceprivate PaymentHandleStrategyFactory paymentHandlerFactory;@GetMapping("/pay")public void pay(int code, double amount) {String paymentType = PaymentEnum.getPaymentNameByCode(code);if (paymentType != null) {// 根据支付方式选择支付处理器IPaymentHandleStrategy paymentHandleStrategy = paymentHandlerFactory.getPaymentHandleStrategy(paymentType);paymentHandleStrategy.pay(amount);}}
}

策略工厂

@Slf4j
@Service
public class PaymentHandleStrategyFactory {/*** 根据支付类型获取相应的处理策略** @param type 支付类型,用于确定使用哪种策略处理支付请求* @return IPaymentHandleStrategy 实现类,用于处理特定支付类型的策略*/public IPaymentHandleStrategy getPaymentHandleStrategy(String type) {// 根据传入的类型参数选择并返回对应的支付处理策略// AliPay+HandleStrategyImplIPaymentHandleStrategy iPaymentHandleStrategy = SpringContextUtil.getBean(type + "HandleStrategyImpl");// 检查 iPaymentHandleStrategy 是否为空,并增加防御性编程if (iPaymentHandleStrategy == null || StringUtils.isEmpty(iPaymentHandleStrategy)) {log.warn("getDatasetHandleStrategy error. type:{}", type);throw new RuntimeException("getDatasetHandleStrategy error. type:" + type);}return iPaymentHandleStrategy;}
}

策略接口

public interface IPaymentHandleStrategy {void pay(double amount);
}

策略实现类

@Service
public class AliPayHandleStrategyImpl implements IPaymentHandleStrategy {@Overridepublic void pay(double amount) {System.out.println("【策略+工厂】支付宝支付:" + amount + "元");}
}@Service
public class WeChatPayHandleStrategyImpl implements IPaymentHandleStrategy {@Overridepublic void pay(double amount) {System.out.println("【策略+工厂】微信支付:" + amount + "元");}
}

枚举类

package com.tyron.design.mode.learn.advanced.enums;public enum PaymentEnum {// 微信支付WE_CHAT_PAY(1, "weChatPay", "微信支付"),// 支付宝支付ALI_PAY(2, "aliPay", "支付宝支付");private int code;private String name;private String desc;PaymentEnum(int code, String name, String desc) {this.code = code;this.name = name;this.desc = desc;}public int getCode() {return code;}public String getName() {return name;}public String getDesc() {return desc;}/*** 根据code获取枚举常量*/public static PaymentEnum getPaymentEnumByCode(int code) {for (PaymentEnum paymentEnum : PaymentEnum.values()) {if (paymentEnum.getCode() == code) {return paymentEnum;}}return null;}/*** 根据code获取枚举常量的name属性** @param code* @return*/public static String getPaymentNameByCode(int code) {for (PaymentEnum paymentEnum : PaymentEnum.values()) {if (paymentEnum.getCode() == code) {return paymentEnum.getName();}}return null;}
}

Spring工具类

package com.tyron.design.mode.learn.util;import lombok.NonNull;
import org.springframework.beans.BeansException;
import org.springframework.context.ApplicationContext;
import org.springframework.context.ApplicationContextAware;
import org.springframework.stereotype.Component;@Component("SpringUtil")
public class SpringContextUtil implements ApplicationContextAware {private static ApplicationContext context;@Overridepublic void setApplicationContext(@NonNull ApplicationContext applicationContext) throws BeansException {context = applicationContext;}@SuppressWarnings("unchecked")public static <T> T getBean(String beanName) {if (context.containsBean(beanName)) {return (T)context.getBean(beanName);} else {return null;}}
}

详细代码已上传 Github:https://github.com/tyronczt/design-mode-learn/tree/main/design-mode-learn-3_2

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【Linux 维测专栏 1 -- Hung Task 分析与验证】

文章目录 Linux Hung Task 简介1. Hung Task 概述2. D 状态与 Hung Task3. Hung Task 的工作原理4. Hung Task 的配置5. Hung Task 的典型输出6. Hung Task 的应用场景7. kernel 配置7.1 编译选项7.2 参数控制7.3 验证方法4. 扩展接口 8. 注意事项 Linux Hung Task 简介 1. Hu…

计算机网络精讲day1——计算机网络的性能指标(上)

性能指标1&#xff1a;速率 概念1&#xff1a;比特 英文全称是binary digit&#xff0c;意思是一个二进制数字&#xff0c;因此一个比特就是二进制数字中的1或0&#xff0c;比特也是信息论中使用的信息量单位。 概念2&#xff1a;速率 网络中的速率指的是数据的传送速率&#…

ubuntu20.04使用matlab2024a快捷键调整

一、概述 因为最近在使用ubuntu系统&#xff0c;在上面安装一个matlab比较方便&#xff0c;不用来回在window上面进行跳转&#xff0c;节省不少时间&#xff0c;同时在ubuntu下面启动matlab速度也比较快。 二、问题解决 &#xff08;一&#xff09;问题概述 问题如下&#xf…

【设计模式】装饰模式

六、装饰模式 装饰(Decorator) 模式也称为装饰器模式/包装模式&#xff0c;是一种结构型模式。这是一个非常有趣和值得学习的设计模式&#xff0c;该模式展现出了运行时的一种扩展能力&#xff0c;以及比继承更强大和灵活的设计视角和设计能力&#xff0c;甚至在有些场合下&am…

科技赋能安全:慧通测控的安全带全静态性能测试

汽车的广泛普及给人们的出行带来了极大便利&#xff0c;但交通事故频发也成为严重的社会问题。据世界卫生组织统计&#xff0c;全球每年约有 135 万人死于道路交通事故&#xff0c;而安全带在减少事故伤亡方面起着不可替代的作用。正确使用安全带可使前排驾乘人员的死亡风险降低…

猿大师中间件:如何在最新Chrome浏览器Web网页内嵌本地OCX控件?

OCX控件是ActiveX控件的一种&#xff0c;主要用于在网页中嵌入复杂的功能&#xff0c;如图形渲染、多媒体播放等&#xff0c;可是随着谷歌Chrome等主流浏览器升级&#xff0c;目前已经不支持微软调用ActiveX控件了&#xff0c;如果想调用OCX控件用IE浏览器或者国产双擎浏览器&a…

基于STC89C52的DS1621温度检测系统

摘要 本文设计了一种基于STC89C52单片机的DS1621温度检测系统&#xff0c;通过DS1621传感器采集温度数据&#xff0c;利用STC89C52单片机处理数据&#xff0c;并通过串口将温度信息发送至电脑显示。借助Proteus软件进行仿真验证&#xff0c;实现了温度检测系统的功能演示。该系…

创建自己的github.io

1、创建GitHub账号 GitHub地址&#xff1a;https://github.com/ 点击Sign up创建账号 如果已创建&#xff0c;点击Sign in登录 2、创建仓库 假设Owner为username&#xff0c;则Repository name为username.github.io说明&#xff1a; 1、Owner为用户名 2、Repository name为仓…

AI Agent中的MCP详解

一、协议定义与核心价值 MCP(Model Context Protocol,模型上下文协议)是由Anthropic公司于2024年11月推出的开放标准协议,其核心目标是通过建立统一接口规范,解决AI模型与外部系统集成效率低下的行业痛点。该协议通过标准化通信机制,使大型语言模型(LLM)能够无缝对接数…

【AI论文】DeepMesh:基于强化学习的自回归艺术家网格创建

摘要&#xff1a;三角形网格在3D应用中扮演着至关重要的角色&#xff0c;能够实现高效的操作和渲染。虽然自回归方法通过预测离散的顶点标记来生成结构化的网格&#xff0c;但它们往往受到面数限制和网格不完整性的约束。为了应对这些挑战&#xff0c;我们提出了DeepMesh框架&a…

基于ArcGIS和ETOPO-2022 DEM数据分层绘制全球海陆分布

第〇部分 前言 一幅带有地理空间参考、且包含海陆分布的DEM图像在研究区的绘制中非常常见&#xff0c;本文将实现以下图像的绘制 关键步骤&#xff1a; &#xff08;1&#xff09;NOAA-NCEI官方下载最新的ETOPO-2022 DEM数据 &#xff08;2&#xff09;在ArcGIS&#xff08;…

Unity | 游戏数据配置

目录 一、ScriptableObject 1.创建ScriptableObject 2.创建asset资源 3.asset资源的读取与保存 二、Excel转JSON 1.Excel格式 2.导表工具 (1)处理A格式Excel (2)处理B格式Excel 三、解析Json文件 1.读取test.json文件 四、相关插件 在游戏开发中,策划…

docker模拟Dos_SYN Flood拒绝服务攻击 (Ubuntu20.04)

目录 ✅ 一、实验环境准备&#xff08;3 个终端&#xff09; &#x1f449; 所以最终推荐做法&#xff1a; 2️⃣ 配置 seed-attacker 为攻击者&#xff0c;开启 telnet 服务&#xff1a; 3️⃣ 配置 victim-10.9.0.5 为受害者服务器&#xff0c;开启 telnet 客户端并监听&…

场外个股期权是什么?场外个股期权还能做吗?

场外个股期权指在非正式的交易场所&#xff0c;即场外市场上&#xff0c;老板们与特定对手方直接进行的个股期权交易。 场外期权为何被严监管&#xff1f; 场外个股期权指在非正式的交易场所&#xff0c;即场外市场上&#xff0c;老板们与特定对手方直接进行的个股期权交易&am…

vulnhub靶场【billu系列】之billu_b0x2靶机

前言 靶机&#xff1a;billu_b0x2靶机&#xff0c;IP地址为192.168.10.10 攻击&#xff1a;kali&#xff0c;IP地址为192.168.10.6 靶机和攻击机都采用VMware虚拟机&#xff0c;都采用桥接网卡模式 文章涉及的靶机及工具&#xff0c;都可以自行访问官网或者项目地址进行获取…

高性能边缘计算网关-高算力web组态PLC网关

高性能EG8200Pro边缘计算算力网关-超强处理能力 样机申请测试&#xff1a;免费测试超30天&#xff08;https://www.iotrouter.com/prototype/&#xff09; 产品主要特点和特色功能 设备概览与连接能力 设备型号&#xff1a;EG8200P。主要特点&#xff1a; 支持多种工业协议&am…

数据可信安全流通实战,隐语开源社区Meetup武汉站开放报名

隐语开源社区 Meetup 系列再出发&#xff01;2025 年将以武汉为始发站&#xff0c;聚焦"技术赋能场景驱动"&#xff0c;希望将先进技术深度融入数据要素流转的各个环节&#xff0c;推动其在实际应用场景中落地生根&#xff0c;助力释放数据要素的最大潜能&#xff01…

避坑指南 | 阿里云服务器centos7上MySQL部署优化指南

目录 1 检查阿里云是否安装mysql 1.1使用 rpm 命令 1.2检查 MySQL 服务状态 2 卸载mysql 2.1停止 MySQL 服务 2.2 检查已安装的 MySQL 包 2.3 卸载 MySQL 包 2.4 删除 MySQL 数据和配置文件 2.5 清理残留的依赖包 2.6 验证卸载 2.7 &#xff08;可选&#xff09;删除…

位运算--求二进制中1的个数

位运算–求二进制中1的个数 给定一个长度为 n 的数列&#xff0c;请你求出数列中每个数的二进制表示中 1 的个数。 输入格式 第一行包含整数 n。 第二行包含 n 个整数&#xff0c;表示整个数列。 输出格式 共一行&#xff0c;包含 n 个整数&#xff0c;其中的第 i 个数表…

Go语言的基础类型

一基础数据类型 一、布尔型&#xff08;Bool&#xff09; 定义&#xff1a;表示逻辑真 / 假&#xff0c;仅有两个值&#xff1a;true 和 false内存占用&#xff1a;1 字节使用场景&#xff1a;条件判断、逻辑运算 二、数值型&#xff08;Numeric&#xff09; 1. 整数类型&…