当前位置: 首页 > news >正文

设计模式-行为型模式-策略模式

1.策略模式的定义

        定义一系列算法,将每个算法封装起来,并使他们可以相互替换,使得算法可以随着使用他的客户端变化而变化;

        当实现某一个功能存在多种算法或策略,我们可以根据环境或者条件的不同选择不同的算法或者策略来实现该功能;

1.1 策略模式的优缺点

优点

  • 由于策略类都实现同一个接口,他们之间可以自由切换;
  • 易于扩展,增加一个新的策略只需要添加一个具体的策略类即可,基本不需要改变原有的代码,符合开闭原则;
  • 避免使用多重条件选择语句(if-else),充分体现面向对象的设计思想;

缺点

  • 客户端必须知道所有的策略类,并自行决定使用哪一个策略类;
  • 策略模式将造成产生很多策略类,可以通过使用享元模式在一定程度上减少对象的数量;

1.2 策略模式的使用场景

  • 一个系统需要动态的在几种算法中选择一种时,可将每个算法封装到策略类中;
  • 一个类定义了多种行为,并且这些行为在这个类的操作中以多个条件语句的形式出现,可将每个条件分支移入他们各自的策略类中以代替这些条件语句;
  • 系统要求使用算法的客户不应该知道其操作的数据时,可使用策略模式来隐藏算法相关的数据结构; 

2.策略模式的原理

  • 抽象策略类(Strategy):通常有一个接口或抽象类实现,给出所有的具体策略类所需接口;
  • 具体策略类(ConcreteStrategy):实现了抽象策略类的接口,提供具体的算法实现或行为;
  • 环境或上下文类(Context):使用算法的角色,持有一个策略类的引用,最终给客户端调用;

3.策略模式的实现

        策略模式的本质就是通过Context作为中心控制单元,对不同的策略进行调度分配;

【实例】

        物流行业中,通常会涉及到EDI报文(XML格式文件)传输和回执接收,每发送一份EDI报文,后续都会收到与之关联的回执(标识该数据在第三方系统中的流转状态);

【代码】

        策略接口

public interface ReceiptHandleStrategy {void handleReceipt(Receipt receipt);
}

        具体策略类

public class Mt1011ReceiptHandleStrategy implements ReceiptHandleStrategy {@Overridepublic void handleReceipt(Receipt receipt) {System.out.println("解析报文MT1011: " + receipt.getMessage());}
}public class Mt2101ReceiptHandleStrategy implements ReceiptHandleStrategy {@Overridepublic void handleReceipt(Receipt receipt) {System.out.println("解析报文MT2101: " + receipt.getMessage());}
}

        策略上下文类

public class ReceiptStrategyContext {private ReceiptHandleStrategy receiptHandleStrategy;public void setReceiptHandleStrategy(ReceiptHandleStrategy receiptHandleStrategy) {this.receiptHandleStrategy = receiptHandleStrategy;}//调用策略类中的方法public void handleReceipt(Receipt receipt){if(receipt != null){receiptHandleStrategy.handleReceipt(receipt);}}
}

        策略工厂

public class ReceiptHandleStrategyFactory {public ReceiptHandleStrategyFactory() {}//使用Map集合存储策略信息,彻底消除if...elseprivate static Map<String,ReceiptHandleStrategy> strategyMap;//初始化具体策略,保存到map集合public static void init(){strategyMap = new HashMap<>();strategyMap.put("MT1011",new Mt1011ReceiptHandleStrategy());strategyMap.put("MT2101",new Mt2101ReceiptHandleStrategy());}//根据回执类型获取对应策略类对象public static ReceiptHandleStrategy getReceiptHandleStrategy(String receiptType){return strategyMap.get(receiptType);}
}

        客户端

public class Client {public static void main(String[] args) {//模拟回执List<Receipt> receiptList = ReceiptBuilder.genReceiptList();//策略上下文ReceiptStrategyContext context = new ReceiptStrategyContext();//策略模式将策略的 定义、创建、使用这三部分进行了解耦for (Receipt receipt : receiptList) {//获取置策略ReceiptHandleStrategyFactory.init();ReceiptHandleStrategy strategy = ReceiptHandleStrategyFactory.getReceiptHandleStrategy(receipt.getType());//设置策略context.setReceiptHandleStrategy(strategy);//执行策略context.handleReceipt(receipt);}}
}

http://www.mrgr.cn/news/20022.html

相关文章:

  • 20240905软考架构-------软考116-120答案解析
  • 10天计划:每天5小时睡眠
  • 【计算机组成原理】你知道什么是8421码、什么是余3码什么又是2421码吗?今天这篇文章带你认识计算机中的BCD码
  • ARP、RARP与路由选择协议
  • Keil下载烧录程序到单片机提示flash outtime超时
  • 支持国产——使用mmdetection进行目标检测并保存推理结果图片
  • vscode任务配置之tasks.json
  • Python | Leetcode Python题解之第385题迷你语法分析器
  • 鸿蒙OS试题
  • ”wait”和“notify”为什么要在Synchronized代码块里面?
  • 用Python实现时间序列模型实战——Day 11: 指数平滑模型
  • matlab 计算3D点到三角面的距离
  • 【基础算法总结】BFS_拓扑排序问题
  • 模电-三极管2
  • 推荐一款开源、高效、灵活的Redis桌面管理工具:Tiny RDM!支持调试与分析功能!
  • 华为OD机试真题 - 停车场车辆统计 - 贪心算法(Java/Python/JS/C/C++ 2024 D卷 200分)
  • 【Rust光年纪】构建高效气象模型计算系统:Rust语言库推荐与比较
  • 浅聊kubernetes 调度
  • 使用Python实现智能金融市场预测
  • 读软件设计的要素01概念