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

news2025/1/16 13:50:31

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

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

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

代码示例

在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和B
        Strategy 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.coloradmin.cn/o/1432630.html

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

相关文章

无人机遥感技术在地质灾害监测应用分析,多旋翼无人机应急救援技术探讨

地质灾害是指在地球的发展演变过程中, 由各种自然地质作用和人类活动所形成的灾害性地质事件。给人民的生命和财产安全带来严重威胁,因此有必要开展地质灾害预测预报、灾害应急和风险区划 遥感技术的快速发展为我们提供了一种获取实时灾害信息的可靠手段…

使用WPS制作三线表

点击边框和底纹点击1、2、3、4并且应用于表格点击确定 再次选中表格点击右键表格属性选择边框和底纹 选中表格第一行右键点击表格属性选择边框和底纹 如果表格中存在虚线

离线数仓-数据治理

目录 一、前言 1.1 数据治理概念 1.2 数据治理目标 1.3 数据治理要解决的问题 1.3.1 合规性 元数据合规性 数据质量合规性 数据安全合规性 1.3.2 成本 存储资源成本 计算资源成本 二、数据仓库发展阶段 2.1 初始期 2.2 扩张期 2.3 缓慢发展期 2.4 变革期 三、…

基于hadoop+spark的大规模日志的一种处理方案

概述: CDN服务平台上有为客户提供访问日志下载的功能,主要是为了满足在给CDN客户提供服务的过程中,要对所有的记录访问日志,按照客户定制的格式化需求以小时为粒度(或者其他任意时间粒度)进行排序、压缩、打包,供客户进行下载,以便进行后续的核对和分析的诉求。而且CDN…

判断和循环 - switch语句和练习

switch语句格式 switch(表达式) {case 值1:语句体1;break;case 值2:语句体2;break;...default:语句体n1;break; }执行流程: 首先计算表达式的值。依次和case后面的值进行比较,如果有对应的值,就会执行相应的语句,在执行的过程中…

vue项目线上页面刷新报404 解决方法

一.修改配置文件 nginx.conf ,并重新加载或重启 我的nginx版本是1.9.9 location / {try_files $uri $uri/ /index.html; }原因: 打包后的dist下只有一个 index.html 文件及一些静态资源,这个是因为Vue是单页应用(SPA),只有一个…

《Python 网络爬虫简易速速上手小册》第6章:Python 爬虫的优化策略(2024 最新版)

文章目录 6.1 提高爬虫的效率6.1.1 重点基础知识讲解6.1.2 重点案例:使用 asyncio 和 aiohttp 实现异步爬虫6.1.3 拓展案例 1:利用 Scrapy 的并发特性6.1.4 拓展案例 2:使用缓存来避免重复请求 6.2 处理大规模数据爬取6.2.1 重点基础知识讲解…

新加坡大带宽服务器优势特点

随着互联网技术的不断进步,大带宽服务器在满足高速数据传输需求方面发挥着越来越重要的作用。新加坡,作为全球互联网基础设施的重要枢纽,其大带宽服务器在全球范围内备受关注。本文将深入探讨新加坡大带宽服务器的优势特点,以及如…

蓝桥杯每日一题----区间dp

前言 暂时没啥好说的,直接进入正题吧 引入 涂色PAINT 读题发现要求的是使一段区间满足要求的最小操作次数,考虑用动态规划去做。 第一步:考虑缩小规模,这里的规模其实就是区间长度,那么dp数组应该可以表示某个区间&…

白话 Transformer 原理-以 BERT 模型为例

白话 Transformer 原理-以 BERT 模型为例 第一部分:引入 1-向量 在数字化时代,数学运算最小单位通常是自然数字,但在 AI 时代,这个最小单元变成了向量,这是数字化时代计算和智能化时代最重要的差别之一。 举个例子:银行在放款前,需要评估一个人的信用度;对于用户而…

解析Python中HTTP代理的常见问题

在Python编程中,HTTP代理是一个经常被提及的概念,尤其在处理网络请求和爬虫时。但与此同时,使用HTTP代理也经常会遇到一些令人头疼的问题。接下来,就让我们一起解析一下Python中使用HTTP代理时常见的那些问题。 1. 代理服务器无响…

06、全文检索 -- Solr -- Solr 全文检索之在图形界面管理 Core 的 Schema(演示对 普通字段、动态字段、拷贝字段 的添加和删除)

目录 Solr 全文检索之管理 Schema使用Web控制台管理Core的Schema3 种 字段解释:Field:普通字段Dynamic Field:动态字段Copy Field:拷贝字段 演示:添加 普通字段( Field )演示:添加 动…

CSS写渐变边框线条

box-sizing: border-box; border-top: 1px solid; border-image: linear-gradient(to right, red, blue) 1;

建筑行业数字化:从设计到运维的全面革新

随着科技的快速发展,数字化技术在各行各业中的应用越来越广泛。建筑行业作为传统产业,也在积极拥抱数字化技术,以提高效率、降低成本并实现可持续发展。本文将主要探讨建筑行业数字化的几个关键领域,包括建筑设计数字化、施工管理…

CDH6.3.2 多 Spark 版本共存

一 部署Spark客户端 1.1 部署spark3客户端 tar -zxvf spark-3.3.1-bin-3.0.0-cdh6.3.2.tgz -C /opt/cloudera/parcels/CDH/lib cd /opt/cloudera/parcels/CDH/lib mv spark-3.3.1-bin-3.0.0-cdh6.3.2/ spark3将 CDH 集群的 spark-env.sh 复制到 /opt/cloudera/parcels/CDH/li…

《Python 网络爬虫简易速速上手小册》第3章:Python 网络爬虫的设计(2024 最新版)

文章目录 3.1 设计高效的爬取策略3.1.1 重点基础知识讲解3.1.2 重点案例:使用 Scrapy 框架进行并发爬取3.1.3 拓展案例 1:使用 Requests 和 gevent 进行异步请求3.1.4 拓展案例 2:利用缓存机制避免重复请求 3.2 管理爬虫的请求频率3.2.1 重点…

【AIGC核心技术剖析】AI生成音乐:MAGNeT一种直接操作多个音频令牌流的掩码生成序列建模方法

MAGNeT是一种直接操作多个音频令牌流的掩码生成序列建模方法。与先前的工作不同,MAGNeT由一个单阶段、非自回归的变压器组成。在训练期间,论文使用掩码调度器预测从掩码令牌中获得的跨度,而在推断期间,论文通过多个解码步骤逐渐构…

微信小程序 使用npm包

1. 微信小程序 使用npm包 1.1. npm初始化 如果你的小程序项目没有安装过npm包的话,你需要先初始化npm npm init1.2. 安装npm包 这里以vant-weapp(小程序UI组件库)为例: npm i vant-weapp -S --production1.3. npm包构建 1.3.1. 点击微信开发者工具右…

怎样理解Vue单向数据流

在前端开发中,数据流是一个非常重要的概念。Vue.js作为一种流行的前端框架,采用了单向数据流的架构,旨在简化开发过程并提高应用的可维护性。本文将探讨Vue单向数据流的含义以及它的使用方法。 什么是单向数据流? 在Vue中&#…

靶机实战bwapp亲测xxe漏洞攻击及自动化XXE注射工具分析利用

靶机实战bwapp亲测xxe漏洞攻击及自动化XXE注射工具分析利用。 1|0介绍 xxe漏洞主要针对webservice危险的引用的外部实体并且未对外部实体进行敏感字符的过滤,从而可以造成命令执行,目录遍历等.首先存在漏洞的web服务一定是存在xml传输数据的,可以在http头的content-type中查…