Spring通过Bean的name实现策略模式
迪丽瓦拉
2025-05-29 04:11:02
0

什么是策略模式?
如果在一个系统里面有许多类,它们之间的区别仅在于它们的行为,那么使用策略模式可以动态地让一个对象在许多行为中选择一种行为。简单理解就是用策略模式替换掉if-else,让代码高内聚低耦合。不同的策略去做不同的事,每个策略之前相互不影响,提高代码的可读性。
策略模式:
优点:
1、算法可以自由切换。
2、避免使用多重条件判断。
3、扩展性良好
缺点:
1、策略类会增多。(我个人觉得不算事缺点吧。人尽其才,物尽其用。)
2、所有策略类都需要对外暴露。(这个才算是缺点,因为我只需要我想要的策略,你把其它我不想要策略都暴露出来了。)
常见的策略模式:Map map = new HashMap<>();这里的Map是策略接口,HashMap是具体实现的策略。Map map = new new TreeMap<>();

场景

假设我不同端的用户和密码都不一样,并且存放到不同的数据库表中,那么按照以前的方法就是通过if-else判断是否是其中的一个端,然后执行代码块。现在我们可以通过策略模式,让前端去调具体端的接口,比如我是web端,那么我前端就传一个web端的具体策略名称参数给后端,后端通过Spring的Bean name获取到具体策略的名称,实现具体的业务,然后把结果返回给前端。这样说可能不太容易理解。

定义策略接口

/*** 1. 新增策略接口。* 应用场景:多端登录,不同的端使用不同的数据库,所以执行不同的策略。*/
public interface Strategy {//2. 定义策略抽象方法(返回类型是T,让你的接口更加通用化!这里的参数你也可以使用String ... str表示传递多个String类型的参数。但是参数嘛,肯定具体的好点。否则可能就需要强转。)T login(String username,String password);
}

实现策略接口

web端

/*** 3.实现web登录策略*/
@Component("webLogin")
public class WebStrategyImpl implements Strategy {@Overridepublic WebTokenInfo login(String username, String password) {//查询web端的数据库,判断用户名和密码是否正确。//正确则返回web端的信息给前端WebTokenInfo webTokenInfo = new WebTokenInfo();webTokenInfo.setWebToken("web端的token");webTokenInfo.setWebUserId(1);return webTokenInfo;}
}

MiniApps端

/*** 3.实现小程序登录策略*/
@Component("miniAppsLogin")
public class MiniAppsStrategyImpl implements Strategy {@Overridepublic MiniAppsTokenInfo login(String username, String password) {//查询web端的数据库,判断用户名和密码是否正确。//正确则返回web端的信息给前端MiniAppsTokenInfo miniAppsTokenInfo = new MiniAppsTokenInfo();miniAppsTokenInfo.setMiniAppsToken("小程序端的token");miniAppsTokenInfo.setMiniAppsUserId(2);return miniAppsTokenInfo;}
}

根据Spring的Bean name去获取前端传递过来的具体策略参数调用具体的策略

/*** 4.根据前端传的bean的名字去执行具体的策略*/
@Component
public class ExecuteStrategy {//strategy就是@Component("name")的name,返回的是个Object更加通用,因为我们无需关注返回对象是什么,早在实现具体的策略的时候我们以及定义好了返回的对象,而Object是所有对象的基类(父类),需要关注Object具体是什么的只有前端!public Object executeStrategy(String strategy, String username, String password) {return  ((Strategy) ApplicationContextUtil.getBean(strategy)).login(username, password);}
}

Controller接收前端传递的策略参数以及登录接口参数

/*** 5.策略模式登录接口,实现多端登录*/
@RestController
@RequestMapping("strategy")
public class StrategyController {@Autowiredprivate ExecuteStrategy executeStrategy;/*** 登录接口* @param strategy 策略名称,即:XXXStrategyImpl的@Component("name")的name。* @param username 用户名(这里是为了策略而策略,所以假设用户名不同端存在不同的数据库中)* @param password 用户密码(这里是为了策略而策略,所以假设密码不同端存在不同的数据库中)*/@GetMapping("{strategy}")//这样写就可以让前端以为调不同的接口,实际上在后端是同一个接口,调的是impl的bean name实现类public Object login(@PathVariable String strategy, @RequestParam("username") String username,@RequestParam("password") String password){//根据前端传递的strategy策略参数,通过Spring的Bean的name去执行具体的策略。return this.executeStrategy.executeStrategy(strategy,username,password);}
}

其它

ApplicationContextUtil工具类

/*** 获取Bean*/
@Component
public class ApplicationContextUtil implements ApplicationContextAware {//构造函数私有化,防止其它人实例化该对象private ApplicationContextUtil(){}private static ApplicationContext applicationContext;@Overridepublic void setApplicationContext(ApplicationContext applicationContext) throws BeansException {ApplicationContextUtil.applicationContext = applicationContext;}//获取applicationContext(一般情况下用户数。因为我们暂时只需要获取bean就不需要获取其它内容了,而获取bean直接调用下面的方法即可。)public static ApplicationContext getApplicationContext() {return applicationContext;}//通过name获取 Bean.(推荐,因为bean的name是唯一的,出现重名的bean启动会报错。)public static Object getBean(String name) {return getApplicationContext().getBean(name);}//通过class获取Bean.(确保bean的name不会重复。因为可能会出现在不同包的同名bean导致获取到2个实例)public static  T getBean(Class clazz) {return getApplicationContext().getBean(clazz);}//通过name,以及Clazz返回指定的Bean(这个是最稳妥的)public static  T getBean(String name, Class clazz) {return getApplicationContext().getBean(name, clazz);}}

MiniAppsTokenInfo小程序返回vo,这个就是上面说的,由于策略接口返回的是个泛型T,我们已经在具体的策略里面定义好了返回类即可。

/*** web端token存储的信息*/
@Data
public class MiniAppsTokenInfo {private String miniAppsToken; //小程序端的tokenprivate Integer miniAppsUserId; //小程序端的用户id
}

WebTokenInfo web端返回vo(同上)

/*** web端token存储的信息*/
@Data
public class WebTokenInfo {private String webToken; //web端的tokenprivate Integer webUserId; //web端的用户id
}

模拟前端调用策略

调用web端的登录策略
在这里插入图片描述
调用小程序端的登录策略
在这里插入图片描述
调用没有的策略,就会抛出异常。
在这里插入图片描述
这里只是模拟,不要纠结为什么在url穿用户名和密码。

相关内容

热门资讯

LinkedList源码解析 Java源码系列:下方连接 http://t.csdn.cn/Nwzed 文章目录...
软件测试2 web测试 (1)web控件测试 ​ 界面检查、单行文本框、多行文本框、...
JVM调优策略 对JVM内存的系统级的调优主要的目的是减少GC的频率和Full GC的次数。   1.Full GC...
快速过一遍ThreadLoca... Thread属性之ThreadLocalMapThreadLocal是java的用来做线程隔离的一个...
【云原生-Docker】Doc... 前面大概介绍了下 Docker组成 Docker大部分的操作都围绕着它的三大核心概念:...
Android---动态权限申... 目录 权限分类 动态权限核心函数 简易实现案例 完整代码     Google 在 Android ...
镜像制作dockerfile编... 1.基于容器制作镜像 示例1: step(1)创建容器并编写内容 [root@...
tcp服务器设置accpet为... 监听socket必须绑定一个端口,以便其他客户端可以连接到这个端口,并与...
试题 历届真题 天干地支【第十... 一、试题来源:第十一届蓝桥杯——天干地支 资源限制 内存限制:256.0...
为什么需要在差分或者重要信号换... 大家可能如果对画PCB没有经验的话,可能不太理解为什么差分线在换层时需要在差分孔旁边打...
Linux线程同步 写在前面 来说线程最后一个内容,今天将补充线程互斥的缺陷,同时我们将学习最常见的一个设计模式,最后我...
快速安装TensorFlow2... 该教程仅适用于初学者,用CPU版本的TensorFlow,安装更快更简单...
从入门到精通:识别滑块验证码缺... 验证码识别是目前互联网应用中普遍存在的技术之一,它通过验证用户输入的信息是否符合要求来...
UART使用 目录 一、uart 二、终端 Terminal 1、终端的分类 2、终端对应的设备节点 三、串口的应...
ONLYOFFICE Docs... ONLYOFFICE Docs crack   文档编辑器   增加了对添加复杂字段的支持。   添...
数学建模-如何用matlab画... 1 画图基本指令hold on :保持打开的命令关闭图形保持功能hold off:title ( x...
ROC曲线和AUC值 ROC曲线(Receiver Operating Characteristic...
【2023.3.8】数据结构复... 【2023.3.8】数据结构复习笔记 文章目录【2023.3.8】数据结构复习笔记序言一、绪论二、线...
一个完整的渗透学习路线是怎样的... 前言 1/我是如何学习黑客和渗透? 我是如何学习黑客和渗透测试的,在这里...
HJ27 查找兄弟单词 描述 定义一个单词的“兄弟单词”为:交换该单词字母顺序(注:...