软件设计模式系列之二十三——策略模式

news2025/1/11 17:55:54

1 模式的定义

策略模式(Strategy Pattern)是一种行为型设计模式,它允许在运行时动态选择算法的行为。这意味着你可以定义一系列算法,将它们封装成独立的策略对象,然后根据需要在不修改客户端代码的情况下切换这些算法。策略模式有助于解决问题领域中不同行为的变化和扩展,同时保持代码的灵活性和可维护性。

2 举例说明

策略模式在日常生活中有许多应用,以下是几个比较符合策略模式且为大家所熟知的例子:

导航应用的路线选择:导航应用根据用户的输入和当前交通情况,选择不同的导航策略,如最短路径、避开拥堵、步行导航等。用户可以根据需要选择不同的导航策略,类似于策略模式中的选择不同算法。
在这里插入图片描述

手机相机的拍摄模式:手机相机应用通常具有多种拍摄模式,如普通拍照、全景模式、夜间模式、慢动作模式等。用户可以根据拍摄需求选择不同的拍摄模式,每种模式对应不同的拍摄策略。

餐厅菜单的点餐方式:在餐厅菜单中,顾客可以选择不同的点餐方式,如套餐、单点、外卖等。每种点餐方式对应不同的计费策略和服务流程。

交通信号灯的控制:交通信号灯会根据不同的时间段和交通流量采用不同的信号控制策略,如红灯、绿灯、黄灯。这些策略会在不同的情况下切换,以保障道路交通的安全和流畅。

这些例子中,策略模式的思想都可以看到:根据不同的需求或条件,选择不同的策略来实现特定的行为或功能,而无需改变核心系统。这种动态选择和切换策略的能力在日常生活中随处可见,帮助我们更好地适应不同的情境和需求。

3 结构

策略模式通常包含以下角色:
在这里插入图片描述

Context(上下文):上下文类负责维护对策略对象的引用,并在运行时切换策略。上下文类通常具有一个方法,用于执行当前策略。

Strategy(策略):策略是一个接口或抽象类,定义了一组算法的共同接口。具体的策略类实现这个接口,提供了不同的算法实现。

ConcreteStrategy(具体策略):具体策略类实现了策略接口,提供了特定算法的实现。

策略模式结构

4 实现步骤

实现策略模式通常包括以下步骤:

定义策略接口:创建一个策略接口或抽象类,定义算法的方法。

创建具体策略类:为每个算法创建具体的策略类,这些类应该实现策略接口,并提供自己的算法实现。

创建上下文类:上下文类应该包含一个策略对象的引用,并提供方法来设置和执行策略。

在客户端代码中使用:在客户端代码中创建上下文对象,并根据需要设置不同的策略。

5 代码实现

让我们通过Java代码来演示策略模式的实现。

// Step 1: 定义策略接口
interface DiscountStrategy {
    double applyDiscount(double amount);
}

// Step 2: 创建具体策略类
class PercentageDiscountStrategy implements DiscountStrategy {
    private double percentage;

    public PercentageDiscountStrategy(double percentage) {
        this.percentage = percentage;
    }

    @Override
    public double applyDiscount(double amount) {
        return amount - (amount * percentage / 100);
    }
}

class FixedAmountDiscountStrategy implements DiscountStrategy {
    private double discountAmount;

    public FixedAmountDiscountStrategy(double discountAmount) {
        this.discountAmount = discountAmount;
    }

    @Override
    public double applyDiscount(double amount) {
        return amount - discountAmount;
    }
}

// Step 3: 创建上下文类
class ShoppingCart {
    private DiscountStrategy discountStrategy;

    public void setDiscountStrategy(DiscountStrategy discountStrategy) {
        this.discountStrategy = discountStrategy;
    }

    public double checkout(double totalAmount) {
        if (discountStrategy != null) {
            return discountStrategy.applyDiscount(totalAmount);
        }
        return totalAmount;
    }
}

// Step 4: 在客户端代码中使用
public class Main {
    public static void main(String[] args) {
        ShoppingCart cart = new ShoppingCart();

        // 使用百分比折扣策略
        cart.setDiscountStrategy(new PercentageDiscountStrategy(10));
        double discountedTotal = cart.checkout(100.0);
        System.out.println("Discounted Total: " + discountedTotal);

        // 使用固定金额折扣策略
        cart.setDiscountStrategy(new FixedAmountDiscountStrategy(20));
        discountedTotal = cart.checkout(100.0);
        System.out.println("Discounted Total: " + discountedTotal);
    }
}

6 典型应用场景

策略模式在许多不同的应用场景中都有广泛的应用。以下是一些典型的策略模式应用场景以及简要说明:

支付系统:支付系统通常需要处理多种支付方式,如信用卡、支付宝、微信支付等。策略模式可以用来实现每种支付方式的处理策略,使系统能够根据用户选择的支付方式执行不同的支付逻辑。
在这里插入图片描述

电商平台的促销策略:电商平台常常会推出不同的促销活动,如打折、满减、赠品等。策略模式可以用来实现这些促销策略,使系统能够根据当前活动动态地计算商品价格。

文本编辑器的文本格式化:文本编辑器可以提供多种文本格式化选项,如加粗、斜体、下划线等。策略模式可以用来实现不同的文本格式化策略,使用户能够选择不同的格式化效果。

游戏开发中的敌人行为:在游戏开发中,不同类型的敌人可能具有不同的行为,如追逐玩家、攻击玩家、躲避玩家等。策略模式可以用来定义不同的敌人行为策略,使游戏引擎能够根据敌人的类型选择正确的行为策略。

数据排序:在数据处理中,可能需要根据不同的排序算法对数据进行排序,如快速排序、冒泡排序、插入排序等。策略模式可以用来实现这些不同的排序策略,使系统能够根据数据量和性质选择合适的排序算法。

这些例子中,策略模式都用于动态选择和切换不同的算法或行为,以满足不同的需求或情境。这种灵活性和可扩展性使得策略模式成为面向对象设计中的重要工具,能够提高代码的可维护性和可读性。

7 优缺点

优点:

策略模式使代码更具灵活性,易于扩展和维护。添加新的策略类不会影响现有代码。
策略模式将算法封装在独立的策略类中,使代码更具可读性和可维护性。
客户端代码可以根据需要在运行时切换策略,无需修改现有代码。

缺点:

如果策略类数量过多,可能会导致类的数量增加,增加维护成本。
客户端需要了解不同策略的存在,可能会增加复杂性。

8 类似模式

与策略模式类似的模式包括以下几种:

状态模式(State Pattern):

状态模式和策略模式都属于行为型设计模式,它们都关注对象的行为,但它们的重点不同。策略模式侧重于在一组算法中选择一个,而状态模式侧重于在对象的状态改变时改变其行为。在状态模式中,对象的行为取决于它的状态,而不是通过外部切换不同的策略类。

命令模式(Command Pattern):

策略模式和命令模式都允许将不同的行为封装成对象,但它们的目的和用途有所不同。策略模式用于在运行时选择不同的算法,而命令模式用于封装请求以及请求的参数,允许以更灵活的方式进行操作。

模板方法模式(Template Method Pattern):

模板方法模式和策略模式都是用于定义一组算法,但它们的关注点不同。模板方法模式定义了一个算法的骨架,具体子类可以实现其中的某些步骤,而策略模式将不同的算法封装成独立的策略对象,并允许在运行时选择不同的策略。

这些模式之间的联系在于它们都涉及到将不同的行为封装成对象,从而提高了代码的可维护性和灵活性。然而,它们各自解决不同类型的问题,因此在选择模式时需要根据具体情况来决定使用哪种模式。策略模式更适用于需要在一组算法中进行动态选择的情况,而其他模式则解决了不同的设计问题,例如状态管理、请求封装等。根据问题的性质,可以选择合适的模式来实现所需的功能。

9 小结

策略模式是一种强大的设计模式,允许在运行时选择不同的算法实现,以处理不同的行为需求。通过将算法封装在独立的策略类中,策略模式提高了代码的可维护性和可扩展性。在面临多种行为选择的情况下,策略模式是一个有力的工具,可以使代码更加灵活和可读。但是,它需要在客户端了解不同策略的情况下使用,因此需要谨慎设计。总之,策略模式是面向对象设计中的一个重要概念,值得深入学习和应用。

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

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

相关文章

POJ 3109 Inner Vertices 离散化+树状数组

一、题目大意 围棋棋盘,如果某个坐标上下左右的四个方向都存在棋子,那么ans1,根据输入的棋子数量,求出ans的数量。 二、解题思路 题目中有说到如果程序不会结束,那么输出-1,这其实是无源之水&#xff0c…

使用Java操作Redis

要在Java程序中操作Redis可以使用Jedis开源工具。 一、jedis的下载 如果使用Maven项目&#xff0c;可以把以下内容添加到pom中 <!-- https://mvnrepository.com/artifact/redis.clients/jedis --> <dependency> <groupId>redis.clients</groupId>…

【Python】函数(function)和方法(method)的区别

这里先说结论&#xff0c;为了满足心急的小伙伴&#xff1a;method与function的最大区别就是参数有无进行绑定。 自定义类Test&#xff1a; 首先先来一个自定义类&#xff1a; class Test:def Func_normal(arg):print(Func_normal:,arg)staticmethoddef Func_static(arg):pri…

在vite中使用react-router-dom-v6 路由报错 Uncaught SyntaxError: Unexpected token ‘<‘

解决方法&#xff1a;将路由表“routes”下面的"index.js"改成“index.jsx”&#xff0c;正确的文件如下图所示。

Springboo整合Sentinel

Springboo整合Sentinel 1.启动Sentinel java -jar sentinel-dashboard-1.8.6.jar2.访问localhost:8080到Sentinel管理界面(默认账号和密码都是sentinel) 3.引入依赖(注意版本对应) <dependency><groupId>com.alibaba.cloud</groupId><artifactId>spr…

window环境下Redis7服务器的安装和运行

一、readis7的下载 由于在官网上没有window版本的redis下载&#xff0c;需要到github中去搜索&#xff0c;以下以redis7为例介绍redis的下载 下载地址&#xff1a;https://github.com/zkteco-home/redis-windows 也可以到百度网盘下载 链接&#xff1a;https://pan.baidu.com…

【JavaSE重点知识归纳】第3节:运算符(算术、关系、逻辑、位、移位、优先级)

目录 一&#xff1a;什么是运算符 二&#xff1a;算术运算符 1.基本四则运算符&#xff1a;加减乘除模&#xff08;、-、*、/、%&#xff09; 2.增量运算符&#xff08;、-、*、%&#xff09; 3.自增/自减运算符&#xff08;、--&#xff09; 三&#xff1a;关系运算符 四…

【Kafka专题】Kafka集群架构设计原理详解

目录 前言前置知识课程内容一、Kafka的Zookeeper元数据梳理1.1 zookeeper整体数据1.2 Controller Broker选举机制1.3 Leader Partition选举机制1.4 Leader Partition自动平衡机制*1.5 Partition故障恢复机制1.6 HW一致性保障-Epoch更新机制1.7 总结 学习总结感谢 前言 Kafka的…

R语言实现竞争风险模型(1)

#竞争风险模型 tmp <- data.frame(gene tiaoxuan[,5:6],OS.Time Train[,"Survival_months"], OS Train[,"CSS"],stringsAsFactors F) colnames(tmp) #方法1&#xff1a;riskregression library(riskRegression) fgr1<-FGR(Hist(OS.Time,OS)~gen…

K 个一组翻转链表(链表反转,固定长度反转)(困难)

优质博文&#xff1a;IT-BLOG-CN 一、题目 给你链表的头节点head&#xff0c;每k个节点一组进行翻转&#xff0c;请你返回修改后的链表。 k是一个正整数&#xff0c;它的值小于或等于链表的长度。如果节点总数不是k的整数倍&#xff0c;那么请将最后剩余的节点保持原有顺序。…

Android SELinux 参数语法介绍及基础分析

Android SELinux安全策略主要使用对象安全上下文的基础进行描述&#xff0c;通过主体和客体的安全上下文去定义主体是否有权限访问客体&#xff0c;称为TypeEnforcement 安全上下文&#xff08;Security Context&#xff09; SEAndroid中的安全上下文:共有4个部分组成分别为use…

STM32 定时器介绍--通用、高级定时器

目录 高级定时器 1.功能框图 1-时钟源 2-时基单元 3-输入捕获 4-输出比较 2.输入捕获的应用 3.输出比较的应用 4.初始化结构体 1-时基初始化结构体 2-输出比较结构体 3-PWM信号 周期和占空比的计算--以通用定时器为例 4-输入捕获结构体 5-断路和死区初始化结构体…

05. 机器学习入门 - 动态规划

文章目录 从一个案例开始动态规划 Hi, 你好。我是茶桁。 咱们之前的课程就给大家讲了什么是人工智能&#xff0c;也说了每个人的定义都不太一样。关于人工智能的不同观点和方法&#xff0c;其实是一个很复杂的领域&#xff0c;我们无法用一个或者两个概念确定什么是人工智能&a…

对一条Linux命令的解读(sed find egrep)

目录 1 sed -i的意义 2 $的作用 3 find . -type f的意义 4 -exec .... {} \;的意义 5 egrep -l的意义 6 综合以上 在前面的博客源码编译elfutils_金色熊族的博客-CSDN博客中&#xff0c;我使用了一条指令 sed -i s/-Werror//g $(find . -type f -exec egrep -l _no_Werr…

学信息系统项目管理师第4版系列15_资源管理基础

1. 项目资源 1.1. 实物资源 1.1.1. 着眼于以有效和高效的方式&#xff0c;分配和使用完成项目所需的实物资源 1.1.2. 包括设备、材料、设施和基础设施 1.2. 团队资源 1.2.1. 人力资源 1.2.2. 包含了技能和能力要求 2. 人力资源管理 2.1. 不仅是组织中最重要的资源之一&…

C语言结构体指针学习

结构体变量存放内存中&#xff0c;也有起始地址&#xff0c;定义一个变量来存放这个地址&#xff0c;那这个变量就是结构体指针&#xff1b; typedef struct mydata{int a1;int a2;int a3; }mydata;void CJgtzzView::OnDraw(CDC* pDC) {CJgtzzDoc* pDoc GetDocument();ASSERT…

【算法分析与设计】回溯法(上)

目录 一、学习要点1.1 回溯法1.2 问题的解空间1.3 0-1背包问题的解空间1.4 旅行售货员问题的解空间1.5 生成问题状态的基本方法 二、回溯法的基本思想三、回溯算法的适用条件四、递归回溯五、迭代回溯六、子集树与排列树七、装载问题八、批处理作业调度问题 一、学习要点 理解回…

Scala第十一章节

Scala第十一章节 1.模式匹配 2. Option 类型 3.偏函数 4.正则表达式 5.异常处理 6.提取器 7.案例&#xff1a;随机职业 scala总目录 文档资料下载

大数据Flink(九十五):DML:Window TopN

文章目录 DML:Window TopN DML:Window TopN Window TopN 定义(支持 Streaming):Window TopN 是一种特殊的 TopN,它的返回结果是每一个窗口内的 N 个最小值或者最大值。 应用场景

【C语言】模拟实现strstr

strstr这个库函数看到这个名字大概率猜不到这是什么函数&#xff0c; 但经过学习就可以很好的认识到这个函数 目录 介绍&#xff1a;模拟实现&#xff1a;思路&#xff1a;代码实现&#xff1a; 介绍&#xff1a; 可以看到此函数是用来寻找一个字符串中是否含有另一个字符串 代…