灵活应对:策略模式在软件设计中的应用

策略模式是一种行为型设计模式,它允许定义一系列算法,并将每个算法封装起来,使它们可以互换使用。策略模式让算法的变化独立于使用算法的客户端,使得在不修改原有代码的情况下切换或扩展新的算法成为可能。

使用策略模式的场景包括但不限于:

当存在多种实现方式,且需要在运行时根据不同条件动态选择具体实现时。例如,一个购物应用可能需要根据用户的会员等级来计算折扣,不同等级对应不同的计算方式,这时就可以使用策略模式来实现。
当存在一组类似的行为,实现细节略有不同,但又不希望通过继承来添加新的子类时。这样可以避免类的爆炸性增长,保持类的单一职责原则。

代码示例

在Java中实现策略模式时,通常会涉及以下几个角色的类和接口,它们之间的关系构成了策略模式的核心:

  1. 上下文(Context):
    维护对策略对象的引用。可定义一个接口来让策略对象访问上下文中的其他数据。
  2. 策略Strategy):
    定义所有支持的算法或行为的策略接口。该接口通常包含一个方法,该方法用于执行策略。
  3. 具体策略(Concrete Strategy):实现策略接口的具体类。每个具体策略类实现算法或行为的一个变体。
  4. 客户端(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),对列表进行排序。
使用流程:

  1. 客户端代码调用Collections.sort()方法,并传入自定义比较器。
  2. Collections.sort()方法接收列表作为参数。
  3. 列表返回其元素给Collections.sort()方法以进行比较。
  4. Collections.sort()方法使用自定义比较器(如果提供)或元素的自然顺序(如果元素实现了Comparable接口)来确定排序逻辑。
  5. 根据选择的策略,Collections.sort()方法对列表进行排序。
    如图所示:
    在这里插入图片描述

以下是 Collections.sort() 方法的策略模式分析:

  1. 策略接口:这里的策略接口是 Comparator,它定义了排序策略的公共行为,即 compare(Object o1, Object o2) 方法。Comparator 可以有多个实现,每个实现提供不同的排序规则。

  2. 具体策略:Comparator 的实现类代表具体策略。例如,Collections.reverseOrder() 返回一个反向排序的比较器,而 Collections.naturalOrder() 返回自然顺序的比较器。用户也可以自定义 Comparator 来表达特定的排序需求。

  3. 上下文(Context):Collections.sort() 方法本身充当上下文角色。它接受一个列表和一个可选的比较器对象。如果提供了比较器,Collections.sort() 会使用该比较器来对列表元素进行排序;如果没有提供,它会使用元素类型的自然顺序(如果元素类型实现了 Comparable 接口)。

  4. 策略的使用:在 Collections.sort() 内部,默认情况下,如果列表元素实现了 Comparable 接口,并且没有提供比较器,那么排序算法将使用 Comparable 接口提供的 compareTo() 方法作为排序策略。如果提供了 Comparator,则使用该比较器的 compare() 方法。

  5. 策略的切换:由于 Collections.sort() 能够接受不同的 Comparator 实现,因此可以在运行时动态地改变排序策略,无需修改排序代码本身。

  6. 算法的独立性:Collections.sort() 方法内部使用了归并排序或者TimSort(Java 7 引入),这个算法独立于策略。策略模式使得算法可以独立于具体的策略实现,增加了代码的灵活性和可扩展性。

优点

  1. 封装性:策略模式通过将算法封装在独立的策略类中,实现了算法与使用算法的客户端之间的解耦,提高了代码的模块性。
  2. 可扩展性:新的策略可以很容易地被添加进系统,符合开闭原则,即对扩展开放,对修改封闭。
  3. 动态替换:可以在运行时根据不同情况选择不同的算法策略,增加了系统的灵活性。
  4. 避免使用多重条件转移:策略模式提供了用组合来替代继承和庞大的条件语句的新思路,有利于代码的维护和理解。

缺点

  1. 系统复杂度:由于策略模式需要定义一系列的策略类,这会增加系统的复杂度。
  2. 数量增多引发的复杂性:随着策略数量的增加,客户端需要进行更多的策略选择和管理,这可能会引入额外的复杂性。
  3. 可能违反单一职责原则:如果策略类承担了过多的职责,或者某些策略实现过于复杂,可能会违背单一职责原则。
    总的来说,策略模式有助于避免使用多重条件语句,随着策略类数量的增长,管理这些策略可能会变得复杂,且在某些情况下可能违反设计原则。因此,在应用策略模式时,应该权衡其带来的灵活性和解耦优势以及可能引入的复杂性和设计问题。

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

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

相关文章

Vivado Tri-MAC IP的例化配置(三速以太网IP)

目录 1 Tri-MAC IP使用RGMII接口的例化配置1.1 Data Rate1.2 interface配置1.3 Shared Logic配置1.4 Features 2 配置完成IP例化视图 1 Tri-MAC IP使用RGMII接口的例化配置 在网络设计中,使用的IP核一般为三速以太网IP核,使用时在大多数场景下为配置为三…

MaxKey 单点登录认证系统——前端后端合并步骤

开发指南 | MaxKey单点登录认证系统 该项目前端是Angular项目,后端是springboot项目 以maxkey-web-app前后端合并为例 构建MaxKey统一认证前端 maxkey-web-frontend/maxkey-web-app ng build --prod --base-href /sign/static/以上sign由以下得来: 根…

Spark Shuffle Service简介与测试

一 Dynamic Resource Allocation(动态资源分配) 了解Shuffle Service之前,我们需要先了解和Shuffle Service有关的另一个特性:动态资源分配。 Spark管理资源有两种方式:静态资源分配和动态资源分配。 静态资源分配:spark提交任…

esp8266 步骤

安装驱动 http://arduino.esp8266.com/stable/package_esp8266com_index.json oled库 esp8266-oled-ssd1306

远程桌面时连接不上远程计算机是什么问题

在服务器上搭建网络程序时,我们经常会有需要远程连接上服务器进行相关操作,有些用户在远程桌面的时候,有时会有遇上无法连接到远程计算机的情况。 很多用户都曾遇到在远程桌面时出现“未启用对服务器的远程访问”、“远程计算机已关闭”、“…

Unix时间戳详解

前言 Unix时间戳最早是在Unix系统中使用的,之后很多由Unix演变而来的系统也都继承了Unix时间戳的规定。目前,Linux、Windows、安卓这些系统的底层计时系统都使用的是Unix时间戳。 一、Unix时间戳简介 Unix时间戳(Unix timestamp&#xff0…

深度学习介绍

对于具备完善业务逻辑的任务,大多数情况下,正常的人都可以给出一个符合业务逻辑的应用程序。但是对于一些包含超过人类所能考虑到的逻辑的任务,例如面对如下任务: 编写一个应用程序,接受地理信息、卫星图像和一些历史…

vscode安装后无右键进入

vscode安装后无右键进入 注:安装时,勾选这两项及可

彻底学会系列:一、机器学习之线性回归

1.基本概念 线性回归: 有监督学习的一种算法。主要关注多个因变量和一个目标变量之间的关系。 因变量: 影响目标变量的因素: X 1 , X 2 . . . X_1, X_2... X1​,X2​... ,连续值或离散值。 目标变量: 需要预测的值: t…

Sklearn、TensorFlow 与 Keras 机器学习实用指南第三版(二)

原文:Hands-On Machine Learning with Scikit-Learn, Keras, and TensorFlow 译者:飞龙 协议:CC BY-NC-SA 4.0 第三章:分类 在第一章中,我提到最常见的监督学习任务是回归(预测值)和分类&#…

jmeter设置定时器

前言 本文主要分享两种定时器(同步定时器、固定定时器)的用法,从作用,设置方法以及实例演示几个方面讲解,结尾还有小知识哦!一起开始学习吧! 一、同步定时器(集合点) …

一个完整工作流管理系统的组成部分

一个完整工作流管理系统的组成部分 一个完整的工作流管理系统通常由工作流引擎、工作流设计器、流程操作、工作流客户端程序、流程监控、表单设计器、与表单的集成以及与应用程序的集成八个部分组成。 一、工作流组成 1. 工作流引擎 工作流引擎作为工作流管理系统的核心部分&…