策略模式在数据接收和发送场景的应用(升级版)

1.背景

在数据接收和发送场景打算使用了 if else 进行判断:


if("A".equals(system)){ASystem.sync("向A同步数据");
}
if("B".equals(system)){BSystem.sync("向B同步数据");
}
...

非常麻烦,需求多了很臃肿

2.策略模式改进

2.1策略模式的定义:​​​​​

策略模式(Strategy Pattern)定义了一组同类型的算法,在不同的类中封装起来,每种算法可以根据当前场景相互替换,从而使算法的变化独立于使用它们的客户端(即算法的调用者)。

2.2策略模式的结构通常包括以下组成部分:

  1. 定义一个策略接口或抽象类:该接口或抽象类定义了所有策略类都需要实现的方法。
  2. 创建多个具体的策略类:每个具体的策略类都实现了策略接口或抽象类,并提供了不同的实现。
  3. 创建一个策略上下文类:该类负责使用策略,它通常会维护一个策略接口或抽象类的引用。
  4. 在客户端代码中使用策略上下文类:客户端代码可以根据需要选择不同的策略。

看定义有些抽象,下面的结构图应该会容易理解一些:

图片

2.3根据上面的结构,我们来实现一下我们的场景。

2.3.1.我们需要定义一个策略接口,定义与外部系统间交互都需要实现的方法

public interface DataProcessingStrategy {void receiveData();void sendData();
}

​​​​2.3.2.为每个外部系统创建一个策略类:

ASystem:

public class ASystemDataProcessingStrategy implements DataProcessingStrategy {@Overridepublic void receiveData() {// 接收数据的具体实现}@Overridepublic void sendData() {// 发送数据的具体实现}
}

BSystem:

public class BSystemDataProcessingStrategy implements DataProcessingStrategy {@Overridepublic void receiveData() {// 接收数据的具体实现}@Overridepublic void sendData() {// 发送数据的具体实现}
}

2.3.3.创建一个选择外部系统的策略类,用于在运行时根据需要选择合适的策略类

public class Context {private DataProcessingStrategy strategy;public Context(DataProcessingStrategy strategy) {this.strategy = strategy;}public void setStrategy(DataProcessingStrategy strategy) {this.strategy = strategy;}public void sendData(String data) {strategy.sendData(data);}public String receiveData() {return strategy.receiveData();}
}

2.3.4.最后,在需要调用外部系统同步数据的地方实例化相关策略类和上下文类,并调用executeStrategy方法:​​​​​​​

public class Main {public static void main(String[] args) {// 创建两个策略对象DataProcessingStrategy strategyA = new ASystemDataProcessingStrategy();DataProcessingStrategy strategyB = new BSystemDataProcessingStrategy();// 创建上下文对象,并传入策略对象Context context = new Context(strategyA);//使用 ASystemDataProcessingStrategy 请求和接收数据context.sendData("");  context.receiveData("");// 使用 BSystemDataProcessingStrategy 请求和接收数据context = new Context(strategyB);context.sendData("");  context.receiveData("");}
}

3.升级为策略模式+工厂模式

那么策略模式存在什么样的问题呢?

  1. 硬编码的依赖关系:在上述代码中,我们直接将具体的策略类(例如StrategyA和StrategyB)硬编码到上下文类(Context)中。这意味着如果我们想要添加或修改策略,我们需要在上下文类中修改代码。这种硬编码的方式使得系统难以扩展和维护。

  2. 客户端与策略的具体实现紧密耦合:由于上下文类Context直接依赖于具体的策略类,因此客户端代码必须了解每个具体策略的细节。这增加了客户端代码的复杂性,并使得客户端代码与策略的具体实现紧密耦合,增加了代码的维护难度。

我们可以使用工厂模式来改进我们的设计。工厂模式可以帮助我们将对象的创建和使用过程分离,使得上下文类和客户端代码不需要了解具体策略的细节,那么我们来修改一下我们的实现:​​​​​​​

context可以去除

// 策略接口和具体的策略类保持不变
public interface DataProcessingStrategy {void sendData(String data);String receiveData();
}public class ASystemDataProcessingStrategy implements DataProcessingStrategy {@Overridepublic void sendData(String data) {// 发送数据到系统A的实现}@Overridepublic String receiveData() {// 从系统A接收数据的实现}
}public class BSystemDataProcessingStrategy implements DataProcessingStrategy {@Overridepublic void sendData(String data) {// 发送数据到系统B的实现}@Overridepublic String receiveData() {// 从系统B接收数据的实现}
}public class DataProcessingStrategyFactory {private static ConcurrentHashMap<String, DataProcessingStrategy> strategies = new ConcurrentHashMap<>();/*** 注册策略* @param strategyName* @param strategy*/public static void register(String strategyName, DataProcessingStrategy strategy) {strategies.put(strategyName, strategy);}public static DataProcessingStrategy getStrategy(String strategyName) {return strategies.get(strategyName);}}//client类相关修改
public class Main {public static void main(String[] args) {DataProcessingStrategy systemA = DeployStrategyFactory.getStrategy("A");//使用 ASystemDataProcessingStrategy 请求和接收数据systemA.sendData("");  systemA.receiveData("");DataProcessingStrategy systemB = DeployStrategyFactory.getStrategy("B");// 使用 BSystemDataProcessingStrategy 请求和接收数据systemB.sendData("");  systemB.receiveData("");}
}

4.总结

在本篇文章中,我们介绍了策略模式,并在数据接收和发送场景中使用了策略模式。通过使用策略模式,我们可以在客户端代码中根据运行时条件动态地选择一个具体的策略类,并通过这个策略类来改变对象的行为。这样,我们就可以实现不同的数据接收和发送方式,而不需要在客户端代码中进行大量的if-else判断。同时通过策略模式+工厂模式的方式解决了客户端代码与策略的具体实现紧密耦合的问题。当然结合实际的场景灵活运用相应的设计模式也非常重要,避免过度设计

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

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

相关文章

SpringBoot实现SSE构建实时数据单向推送

SSE 是一种单向通信&#xff0c;只允许服务器向客户端发送数据。客户端无法向服务器发送数据。SSE 建立在 HTTP 协议之上&#xff0c;使用标准 HTTP 请求和响应。SSE 不需要额外的库或协议处理&#xff0c;客户端可以使用浏览器的原生 EventSource API 来接收数据。SSE 支持跨域…

C#,数值计算——插值和外推,分段线性插值(Linear_interp)的计算方法与源程序

1 文本格式 using System; namespace Legalsoft.Truffer { /// <summary> /// 分段线性插值 /// Piecewise linear interpolation object. /// Construct with x and y vectors, then call interp for interpolated values. /// </summary> …

Windows 安装 Docker

目录 前言安装 WSL2WSL2 简介系统要求安装步骤 安装 Docker Desktop下载安装验证 安装 Docker Compose结语开源项目 前言 下图展示了在 Windows 系统上安装 Docker&#xff0c;并利用Docker Compose一键搭建 youlai-mall 微服务商城所需的环境。本篇将先介绍 Windows 上如何安…

【Linux】指令详解(二)

目录 1. 前言2. 重新认识指令2.1 指令的本质2.1.1 which2.1.2 alias 3. 常见指令3.1 whoami3.2 cd3.2.1 cd -3.2.2 cd ~ 3.3 touch3.3.1 文件创建时间 3.4 stat3.5 mkdir3.5.1 创建一个目录3.5.2 创建路径 3.6 tree3.7 rm3.7.1 rm -f3.7.2 rm -r 3.8 man3.9 cp3.10 mv 1. 前言 …

Leetcode刷题详解——删除并获得点数

1. 题目链接&#xff1a;740. 删除并获得点数 2. 题目描述&#xff1a; 给你一个整数数组 nums &#xff0c;你可以对它进行一些操作。 每次操作中&#xff0c;选择任意一个 nums[i] &#xff0c;删除它并获得 nums[i] 的点数。之后&#xff0c;你必须删除 所有 等于 nums[i] …

Git 版本控制工具

目录 一、集中式版本控制和分布式版本控制的区别 二、Bash - CMD - GUI 一、集中式版本控制和分布式版本控制的区别 SVN 是集中式版本控制工具&#xff0c;它会将所有的内容存储到一台服务器上&#xff0c;用户通过对服务器中的内容进行操作&#xff0c;从而获取最新的内容。…

#gStore-weekly | gBuilder功能详解之结构化数据抽取

上一个weekly中已经详细讲解了schema的设计&#xff0c;在schema设计好了之后&#xff0c;gBuilder支持将结构化和非结构化数据转化为RDF图数据。其中结构化数据支持数据的无损转化。 1. 技术介绍 gBuilder的结构化数据抽取采用D2RQ技术实现。 DR2Q是一个能够将关系数据库中…

linux上安装qt creator

linux上安装Qt Creator 1 Qt Creator 的下载 下载地址为&#xff1a;http://download.qt.io/archive/qt/ 根据自己的需求选择Qt Creator版本&#xff0c;这里我下载的是5.12.9&#xff0c;如下图所示&#xff1a; 在ubuntu上可以使用wget命令下载安装包&#xff1a; wget h…

【如何学习Python自动化测试】—— 浏览器操作

4 、 浏览器操作 4.1 浏览器最大化 Webdriver 打开浏览器后&#xff0c;默认不是最大化&#xff0c;如果需要界面最大化&#xff0c;需要通过 maximize_window()方法来实现&#xff0c;代码如下&#xff1a; maximize_window()方法是Selenium WebDriver提供的一个方法&#xf…

Error message “error:0308010C:digital envelope routines::unsupported“

1.降级到 Node.js v16。 您可以从 Node.js 的 website 重新安装当前的 LTS 版本。 您也可以使用 nvm。对于 Windows&#xff0c;请使用 nvm-windows。 2.启用传统 OpenSSL 提供程序。 在类 Unix 系统&#xff08;Linux、macOS、Git bash 等&#xff09;上&#xff1a; exp…

【Qt开发流程】之程序主窗口

描述 就目前的应用程序而言&#xff0c;一般包含菜单栏、工具栏、状态栏、中央区域等。 qt窗口部件类图如下&#xff1a; 一个主窗口提供了一个构建应用程序用户界面的框架。 Qt有QMainWindow及其相关类来管理主窗口。 QMainWindow有自己的布局&#xff0c;可以向其中添加QTo…

Stable Diffusion 秋葉aaaki整合包远程访问设置

Stable Diffusion 秋葉aaaki整合包远程访问设置 0. 背景1. 解决方法 12. 解决方法 2 0. 背景 在局域网的一台服务器上安装了秋葉aaaki整合包&#xff0c;实现局域网内其他机器访问这台服务器上启动的 Stable Diffusion Web UI&#xff0c;但是默认的启动 server_name 是 127.0…

归并排序知识总结

归并排序思维导图&#xff1a; 知识点&#xff1a;如果原序列中两个数的值是相同的&#xff0c;它们在排完序后&#xff0c;它们的位置不发生变化&#xff0c;那么这个排序是稳定的。快速排序是不稳定的&#xff0c;归并排序是稳定的。 快排变成稳定的>使快排排序数组中的每…

矩阵的模和内积

模和内积 向量 设存在一个向量 X { x 1 , x 2 , x 3 … x n } T X\{x_1,x_2,x_3\dots x_n\}^T X{x1​,x2​,x3​…xn​}T P范数 ∣ ∣ X ∣ ∣ P ∑ i 1 n ∣ x i ∣ p p ||X||_P\sqrt[p]{\sum_{i1}^{n}{|x_i|}^p} ∣∣X∣∣P​pi1∑n​∣xi​∣p ​ 1范数&#xff08;曼…

Scala的一等公民和至简原则

1. Scala 中&#xff0c;函数是一等公民具体体现在哪里 Scala 混合了面向对象特性和函数式的特性函数可以作为值传递&#xff1a;函数可以作为参数传递给其他函数&#xff0c;也可以作为返回值返回给其他函数函数可以赋值给变量&#xff1a;和其他数据类型⼀样&#xff0c;函数…

24 - 内存持续上升,我该如何排查问题?

我想你肯定遇到过内存溢出&#xff0c;或是内存使用率过高的问题。碰到内存持续上升的情况&#xff0c;其实我们很难从业务日志中查看到具体的问题&#xff0c;那么面对多个进程以及大量业务线程&#xff0c;我们该如何精准地找到背后的原因呢&#xff1f; 1、常用的监控和诊断…

Python - Wave2lip 环境配置与 Wave2lip x GFP-GAN 实战 [超详细!]

一.引言 前面介绍了 GFP-GAN 的原理与应用&#xff0c;其用于优化图像画质。本文关注另外一个相关的项目 Wave2lip&#xff0c;其可以通过人物视频与自定义音频进行适配&#xff0c;改变视频中人物的嘴型与音频对应。 二.Wave2Lip 简介 Wave2lip 研究 lip-syncing 以达到视频…

基于Springboot的地方美食分享网站(有报告)。Javaee项目,springboot项目。

演示视频&#xff1a; 基于Springboot的地方美食分享网站(有报告)。Javaee项目&#xff0c;springboot项目。 前些天发现了一个巨牛的人工智能学习网站&#xff0c;通俗易懂&#xff0c;风趣幽默&#xff0c;忍不住分享一下给大家。点击跳转到网站。 项目介绍&#xff1a; 采用…

常见树种(贵州省):009楠木、樟木、桂木种类

摘要&#xff1a;本专栏树种介绍图片来源于PPBC中国植物图像库&#xff08;下附网址&#xff09;&#xff0c;本文整理仅做交流学习使用&#xff0c;同时便于查找&#xff0c;如有侵权请联系删除。 图片网址&#xff1a;PPBC中国植物图像库——最大的植物分类图片库 一、楠木 …

2.Spring的优缺点是什么?

Spring的优缺点是什么 特点&#xff1a;1.方便解耦&#xff0c;简化开发2.AOP编程的支持3.声明事物的支持4.方便程序的测试5.方便集成各种优秀框架6.降低Java EE API的使用难度7.Java 源码是经典学习范例 缺点 特点&#xff1a; 1.方便解耦&#xff0c;简化开发 通过Spring提…