策略模式的实现与应用:掌握灵活算法切换的技巧

文章目录

  • 常用的设计模式有以下几种:
    • 一.创建型模式(Creational Patterns):
    • 二.结构型模式(Structural Patterns):
    • 三.行为型模式(Behavioral Patterns):
    • 四.并发模式(Concurrent Patterns):
  • 策略模式概念
  • 一、策略模式例子一
  • 二、策略模式例子二
  • 三、策略模式例子三
  • 总结


常用的设计模式有以下几种:

一.创建型模式(Creational Patterns):

工厂模式(Factory Pattern)
抽象工厂模式(Abstract Factory Pattern)
单例模式(Singleton Pattern)
原型模式(Prototype Pattern)
建造者模式(Builder Pattern)

二.结构型模式(Structural Patterns):

适配器模式(Adapter Pattern)
桥接模式(Bridge Pattern)
装饰器模式(Decorator Pattern)
组合模式(Composite Pattern)
外观模式(Facade Pattern)
享元模式(Flyweight Pattern)
代理模式(Proxy Pattern)

三.行为型模式(Behavioral Patterns):

策略模式(Strategy Pattern)
观察者模式(Observer Pattern)
迭代器模式(Iterator Pattern)
命令模式(Command Pattern)
模板方法模式(Template Method Pattern)
职责链模式(Chain of Responsibility Pattern)
状态模式(State Pattern)
访问者模式(Visitor Pattern)
中介者模式(Mediator Pattern)
备忘录模式(Memento Pattern)
解释器模式(Interpreter Pattern)

四.并发模式(Concurrent Patterns):

单例模式(Singleton Pattern,在多线程环境下的应用)
生成器模式(Builder Pattern,在多线程环境下的应用)
保护性暂停模式(Guarded Suspension Pattern)
观察者模式(Observer Pattern,在多线程环境下的应用)

这些设计模式涵盖了对象的创建、对象间的组织和交互、以及对象的行为等方面,可以帮助开发者更好地组织和设计代码,提高代码的可扩展性、可维护性以及重用性。需要根据实际情况选择适合的设计模式来解决问题。


策略模式概念

今天来讲一讲策略模式:
策略模式(Strategy Pattern)是一种行为设计模式,它允许你定义一系列算法,并将每个算法封装到独立的类中,使它们可以互相替换。策略模式使得算法的变化独立于使用它们的客户端。
在策略模式中,有一个上下文(Context)对象,它持有一个策略接口的引用,并通过调用策略接口的方法来执行具体的算法。算法类实现了策略接口,每个算法类都封装了一种具体的算法,客户端可以根据需要选择不同的策略对象,从而实现不同的行为。🤩🤩🤩
策略模式的优点包括:

1. 算法可以灵活地替换或新增,不影响使用算法的客户端。
2. 可以消除大量的条件语句,提高代码的可维护性和可读性。
3. 可以将算法的实现细节封装起来,提高代码的模块性和复用性。

策略模式在许多场景下都有应用,比如排序算法、计算费用、验证用户等。通过使用策略模式,可以优雅地处理复杂的逻辑,并提供一种可扩展、易于维护的解决方案。🤗🤗🤗

我并不喜欢看含义,对于这种概念性的东西,动手实践才是硬道理!
以下举出 策略模式 的几个 荔枝!!!😍😍😍
相信看完会更加理解策略模式的强大性!

在这里插入图片描述

一、策略模式例子一

假设我们正在开发一个商场收银系统,其中有不同的折扣策略供用户选择,包括无折扣、固定折扣和打折折扣。我们希望通过策略模式来实现这些不同的折扣策略。
首先,我们定义一个抽象的策略接口 DiscountStrategy,它声明了一个计算折扣金额的方法 calculateDiscount:

public interface DiscountStrategy {double calculateDiscount(double amount);
}

然后,我们分别实现三种具体的策略类:NoDiscountStrategy(无折扣)、FixedDiscountStrategy(固定折扣)和PercentageDiscountStrategy(打折折扣)。它们都实现了 DiscountStrategy 接口:

public class NoDiscountStrategy implements DiscountStrategy {@Overridepublic double calculateDiscount(double amount) {return 0; // 无折扣,返回0}
}public class FixedDiscountStrategy implements DiscountStrategy {private double discount;public FixedDiscountStrategy(double discount) {this.discount = discount;}@Overridepublic double calculateDiscount(double amount) {return discount; // 固定折扣,直接返回固定的折扣金额}
}public class PercentageDiscountStrategy implements DiscountStrategy {private double percentage;public PercentageDiscountStrategy(double percentage) {this.percentage = percentage;}@Overridepublic double calculateDiscount(double amount) {return amount * percentage; // 打折折扣,根据折扣比例计算折扣金额}
}

最后,我们定义一个上下文类 Cashier,它持有一个 DiscountStrategy 的引用,并在结算时调用该策略的方法:

public class Cashier {private DiscountStrategy discountStrategy;public void setDiscountStrategy(DiscountStrategy discountStrategy) {this.discountStrategy = discountStrategy;}public double calculateTotalWithDiscount(double amount) {double discount = discountStrategy.calculateDiscount(amount);return amount - discount;}
}

现在,我们可以使用策略模式来实现商场收银系统的折扣功能了。示例如下:

public class Main {public static void main(String[] args) {Cashier cashier = new Cashier();// 设置无折扣策略DiscountStrategy noDiscountStrategy = new NoDiscountStrategy();cashier.setDiscountStrategy(noDiscountStrategy);double total = cashier.calculateTotalWithDiscount(100);System.out.println("Total with no discount: " + total);// 设置固定折扣策略DiscountStrategy fixedDiscountStrategy = new FixedDiscountStrategy(20);cashier.setDiscountStrategy(fixedDiscountStrategy);total = cashier.calculateTotalWithDiscount(100);System.out.println("Total with fixed discount: " + total);// 设置打折折扣策略DiscountStrategy percentageDiscountStrategy = new PercentageDiscountStrategy(0.1);cashier.setDiscountStrategy(percentageDiscountStrategy);total = cashier.calculateTotalWithDiscount(100);System.out.println("Total with percentage discount: " + total);}
}

输出结果:

Total with no discount: 100.0
Total with fixed discount: 80.0
Total with percentage discount: 90.0

通过策略模式,我们可以灵活地选择不同的折扣策略,并在运行时动态地切换和应用这些策略,而不需要修改 Cashier 类的代码。这样,我们可以方便地添加新的折扣策略或修改现有策略的实现,而不会对其他部分产生影响。这样的设计符合开闭原则,增加了代码的可维护性和扩展性。

二、策略模式例子二

例子2:排序算法
假设我们正在设计一个排序工具,它可以根据用户选择的不同排序算法对数据进行排序。我们可以使用策略模式来实现不同的排序策略。
首先,我们定义一个排序策略接口 SortingStrategy,它声明了一个排序方法 sort:

public interface SortingStrategy {void sort(int[] array);
}

然后,我们分别实现两种具体的策略类:BubbleSortStrategy(冒泡排序)和 QuickSortStrategy(快速排序)。它们都实现了 SortingStrategy 接口:

public class BubbleSortStrategy implements SortingStrategy {@Overridepublic void sort(int[] array) {// 冒泡排序算法的具体实现// ...}
}public class QuickSortStrategy implements SortingStrategy {@Overridepublic void sort(int[] array) {// 快速排序算法的具体实现// ...}
}

最后,我们定义一个上下文类 Sorter,它持有一个 SortingStrategy 的引用,并在排序时调用该策略的方法:

public class Sorter {private SortingStrategy sortingStrategy;public void setSortingStrategy(SortingStrategy sortingStrategy) {this.sortingStrategy = sortingStrategy;}public void sortArray(int[] array) {sortingStrategy.sort(array);}
}

现在,我们可以使用策略模式来实现排序工具了。示例如下:

public class Main {public static void main(String[] args) {Sorter sorter = new Sorter();// 使用冒泡排序策略SortingStrategy bubbleSortStrategy = new BubbleSortStrategy();sorter.setSortingStrategy(bubbleSortStrategy);int[] array = {5, 3, 8, 2, 1};sorter.sortArray(array);System.out.println("Sorted array using bubble sort:");for (int num : array) {System.out.print(num + " ");}System.out.println();// 使用快速排序策略SortingStrategy quickSortStrategy = new QuickSortStrategy();sorter.setSortingStrategy(quickSortStrategy);int[] array2 = {9, 4, 6, 2, 7};sorter.sortArray(array2);System.out.println("Sorted array using quick sort:");for (int num : array2) {System.out.print(num + " ");}System.out.println();}
}

输出结果:

Sorted array using bubble sort:
1 2 3 5 8 
Sorted array using quick sort:
2 4 6 7 9

通过策略模式,我们可以灵活地选择不同的排序策略,而不需要修改 Sorter 类的代码。这样,我们可以方便地添加新的排序策略或修改现有策略的实现,而不会对其他部分产生影响。

三、策略模式例子三

假设我们正在开发一个电商平台,针对不同的用户类型(普通用户、VIP用户、管理员),我们需要为其提供不同的登录方式。
首先,我们定义一个登录策略接口 LoginStrategy,它声明了一个登录方法 login:

public interface LoginStrategy {void login();
}

然后,我们分别实现三种具体的策略类:DefaultLoginStrategy(普通用户登录方式)、VIPLoginStrategy(VIP用户登录方式)和 AdminLoginStrategy(管理员登录方式)。它们都实现了 LoginStrategy 接口:

public class DefaultLoginStrategy implements LoginStrategy {@Overridepublic void login() {System.out.println("使用用户名和密码登录");// 具体的登录逻辑}
}public class VIPLoginStrategy implements LoginStrategy {@Overridepublic void login() {System.out.println("使用手机号和密码登录");// 具体的登录逻辑}
}public class AdminLoginStrategy implements LoginStrategy {@Overridepublic void login() {System.out.println("使用管理员账号和密码登录");// 具体的登录逻辑}
}

最后,我们定义一个上下文类 LoginContext,它持有一个 LoginStrategy 的引用,并在登录时调用该策略的方法:

public class LoginContext {private LoginStrategy loginStrategy;public void setLoginStrategy(LoginStrategy loginStrategy) {this.loginStrategy = loginStrategy;}public void performLogin() {loginStrategy.login();}
}

现在,我们可以使用策略模式来为不同的用户类型提供不同的登录方式了。示例如下:

public class Main {public static void main(String[] args) {LoginContext loginContext = new LoginContext();// 普通用户登录LoginStrategy defaultLoginStrategy = new DefaultLoginStrategy();loginContext.setLoginStrategy(defaultLoginStrategy);loginContext.performLogin();// VIP用户登录LoginStrategy vipLoginStrategy = new VIPLoginStrategy();loginContext.setLoginStrategy(vipLoginStrategy);loginContext.performLogin();

总结

UU们,懂了嘛!!!😄😄😄
以下是实际开发中常用的几个例子,展示了策略模式的应用:

支付方式:
在电商平台上,用户可以选择不同的支付方式进行支付,比如支付宝、微信支付、银行卡支付等。这些支付方式都具备支付的功能,但其实现方式和支付逻辑可能有所不同。策略模式可以用于将不同的支付方式封装成不同的策略类,然后在运行时根据用户选择的支付方式来选择相应的策略进行支付。

缓存策略:
在系统开发中,我们经常会使用缓存来提高系统性能。不同的数据操作可能需要不同的缓存策略,比如基于时间的过期策略、基于LRU(Least
Recently Used)的淘汰策略、基于LFU(Least Frequently
Used)的淘汰策略等。策略模式可以用于封装不同的缓存策略为不同的策略类,然后在运行时选择相应的缓存策略进行数据操作。

图像处理:
在图像处理应用中,我们可能需要应用不同的滤镜或特效来处理图像,比如黑白滤镜、模糊滤镜、锐化滤镜等。这些滤镜都具备处理图像的功能,但实现方式和处理逻辑可能有所不同。策略模式可以用于将不同的滤镜或特效封装成不同的策略类,然后在运行时根据用户选择的滤镜或特效来选择相应的策略进行图像处理。

日志记录级别:
在日志记录应用中,不同的日志级别对应不同的日志输出方式,比如普通日志输出到控制台,警告日志输出到文件,错误日志发送邮件等等。策略模式可以用于封装不同的日志记录级别为不同的策略类,然后在运行时根据日志级别选择相应的策略进行日志记录。

这些都是在实际开发中常用的例子,策略模式可以帮助我们根据不同的条件和需求来选择相应的策略进行处理,提供了灵活性和扩展性。它可以将不同的策略封装起来,使客户端代码与具体策略的实现解耦,从而提高代码的可维护性和可测试性。
(注意:是不同的处理逻辑,如果你只是根据不同类型查数据库一张表的数据,那就没啥必要了)
不懂或者有什么错误的地方,欢迎评论区留言,俺可以再举更多的 荔枝~~~

@作者:加辣椒了吗?
简介:憨批大学生一枚,喜欢在博客上记录自己的学习心得,也希望能够帮助到你们!
在这里插入图片描述

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

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

相关文章

Spring注解系列——@PropertySource

在Spring框架中PropertySource注解是非常常用的一个注解,其主要作用是将外部化配置解析成key-value键值对"存入"Spring容器的Environment环境中,以便在Spring应用中可以通过Value或者占位符${key}的形式来使用这些配置。 使用案列 // Propert…

基于Citespace、vosviewer、R语言的文献计量学可视化分析及SCI论文高效写作方法教程

详情点击链接:基于Citespace、vosviewer、R语言的文献计量学可视化分析技术及全流程文献可视化SCI论文高效写作方法 前言 文献计量学是指用数学和统计学的方法,定量地分析一切知识载体的交叉科学。它是集数学、统计学、文献学为一体,注重量…

工业边缘计算为什么?

在工厂环境中使用边缘计算并不新鲜。可编程逻辑控制器(PLC)、微控制器、服务器和PC进行本地数据处理,甚至是微型数据中心都是边缘技术,已经在工厂系统中存在了几十年。在车间里看到的看板系统,打卡系统,历史…

Flowable中间事件-中间信号捕获事件

定义 信号中间事件分为 Catching 事件和 Throwing 事件,即信号中间捕获事件(Signal Intermediate Catch Event)和信号中间抛出事件(Signal Intermediate Throwing Event)。 当流程执行到信号中间捕获事件时就会中断在…

什么是ROC曲线

ROC曲线(Receiver Operating Characteristic Curve),也称为“接受者操作特性曲线”。它最早应用于雷达信号检测的分析,后来广泛应用于心理学和医学领域。 ROC分析是进行临床诊断试验评价最常用的方法。诊断试验是指评价某种疾病诊…

Xshell配置ssh免密码登录-公钥与私钥登录linux服务器

目录 简介 提示 方法步骤 步骤1:生成密钥公钥(Public key)与私钥(Private Key) 方法1:使用xshell工具 方法2:使用命令行 步骤2:放置公钥(Public Key)到服务器 方法1:(我使用的是…

InnoDB数据存储结构

一. InnoDB的数据存储结构:页 索引是在存储引擎中实现的,MySQL服务器上的存储引擎负责对表中数据的读取和写入工作。不同存储引擎中存放的格式一般不同的,甚至有的存储引擎比如Memory都不用磁盘来存储数据,这里讲讲InooDB存储引擎…

NineData支持最受欢迎数据库PostgreSQL

根据在 Stack Overflow 发布的 2023 开发者调研报告中显示,PostgreSQL 以 45% vs 41% 的受欢迎比率战胜 MySQL,成为新的最受欢迎的数据库。NineData 也在近期支持了 PostgreSQL,用户可以在 NineData 平台上进行创建数据库/Schema、管理用户与…

解决AttributeError: ‘DataParallel‘ object has no attribute ‘xxxx_fc1‘

问题描述 训练模型时,分阶段训练,第二阶段加载第一阶段训练好的模型的参数,接着训练 第一阶段训练,含有代码 if (train_on_gpu):if torch.cuda.device_count() > 1:net nn.DataParallel(net)net net.to(device)第二阶段训练…

Linux环境Arduino IDE中配置ATOM S3

linux选择ubuntu发行版。 硬件设备有多小呢: 功能超级强大。 之前的ROS1和ROS2案例已经全部移植完成并测试结束(三轮纯人力校验😎)。 官网文档信息非常非常好: https://docs.m5stack.com/zh_CN/quick_start/atoms3…

Jenkins 配置maven和jdk

前提:服务器已经安装maven和jdk 一、在Jenkins中添加全局变量 系统管理–>系统配置–>全局属性–>环境变量 添加三个全局变量 JAVA_HOME、MAVEN_HOME、PATH 二、配置maven 系统管理–>全局工具配置–>maven–>新增 新增配置 三、配置JDK 在系统管…

科普 | OSI模型

本文简要地介绍 OSI 模型 1’ 2’ 3。 更新:2023 / 7 / 23 科普 | OSI模型 术语节点链路协议网络拓扑 概念作用结构应用层表示层会话层传输层网络层数据链路层物理层 数据如何流动OSI 和TCP/IP 的对应关系和协议参考链接 术语 节点 节点( Node &#…

Go 语言切片是如何扩容的?

在 Go 语言中,有一个很常用的数据结构,那就是切片(Slice)。 切片是一个拥有相同类型元素的可变长度的序列,它是基于数组类型做的一层封装。它非常灵活,支持自动扩容。 切片是一种引用类型,它有…

前端(九)——探索微信小程序、Vue、React和Uniapp生命周期

🙂博主:小猫娃来啦 🙂文章核心:探索微信小程序、Vue、React和Uniapp生命周期 文章目录 微信小程序、Vue、React和Uniapp的基本定义和应用领域微信小程序生命周期生命周期概述页面生命周期应用生命周期组件和API的生命周期钩子 Vu…

【雕爷学编程】MicroPython动手做(16)——掌控板之图片图像显示

知识点:什么是掌控板? 掌控板是一块普及STEAM创客教育、人工智能教育、机器人编程教育的开源智能硬件。它集成ESP-32高性能双核芯片,支持WiFi和蓝牙双模通信,可作为物联网节点,实现物联网应用。同时掌控板上集成了OLED…

基于注解的 SpringMVC

SpringMVC SpringMVC使用SpringMVC的两个配置EnableWebMVC 和 ACWACSpringMVC执行流程接收请求参数Postman 发包工具()get 请求---简单类型数据(基本数据类型和String)get 请求---对象类型数据get 请求---数组类型get 请求 --- 集…

Python自动计算Excel数据指定范围内的区间最大值

本文介绍基于Python语言,基于Excel表格文件内某一列的数据,计算这一列数据在每一个指定数量的行的范围内(例如每一个4行的范围内)的区间最大值的方法。 已知我们现有一个.csv格式的Excel表格文件,其中有一列数据&#…

基于光子实验的指数级加速的量子同态加密理论

前言 量子计算机不仅有望在某些重要任务上超越经典计算机,而且还能保护计算的隐私。例如,盲量子计算协议支持安全委托量子计算,其中客户端可以保护其数据和算法的隐私,不受分配来运行计算的量子服务器的影响。然而,这…

【雕爷学编程】MicroPython动手做(14)——掌控板之OLED屏幕2

知识点:什么是掌控板? 掌控板是一块普及STEAM创客教育、人工智能教育、机器人编程教育的开源智能硬件。它集成ESP-32高性能双核芯片,支持WiFi和蓝牙双模通信,可作为物联网节点,实现物联网应用。同时掌控板上集成了OLED…

Visual Studio2022报错 无法打开 源 文件 “openssl/conf.h“解决方式

目录 问题起因问题解决临时解决方案 问题起因 近一段时间有了解到Boost 1.82.0新添加了MySQL库,最近一直蠢蠢欲动想要试一下这个库 所以就下载了源码并进行了编译(过程比较简单,有文档的) 然后在VS2022中引入了Boost环境&#xf…