[设计模式]springboot优雅实现策略器模式(加入注册器实现)

2023-09-17 14:12:46

场景

登陆场景使用(登陆之后返回用户信息和token所做操作基本一致,杜绝多个判断)

  1. 用户名密码登陆,
  2. 指纹登录
  3. 手机号登陆
    楼主之前写过一次通过注解实现的,那个看起来没有注册器实现优雅。
    策略器可以使用的场景很多,主要为了杜绝多if判断,把if判断需实现的信息放在自己的实现类中,提升代码可阅读性,提高执行效率,降低出错概率。
    八股文:可扩展性高,灵活性高,代码复用性高,单一职责原则,可替换性高。

废话少说,源码地址。

代码地址 
git clone https://github.com/gwy572294624/strategy-demo-new


讲解

chatGpt关于策略模式的八股文
策略模式(Strategy Pattern)是一种行为型设计模式,用于将不同的算法封装成独立的策略类,并使这些策略类可以相互替换,以实现在运行时动态地选择不同的算法。

在策略模式中,有三个主要的角色:

  1. 环境(Context):环境类是策略模式的核心,其内部持有一个策略对象的引用。环境类将具体的任务委派给策略对象进行处理,而不直接实现具体的算法逻辑。

  2. 策略(Strategy):策略类是一个接口或抽象类,它定义了具体算法的方法。不同的策略类实现了不同的算法逻辑。策略类之间可以相互替换,提供了算法的灵活配置。

  3. 具体策略(Concrete Strategies):具体策略类是策略模式的实现类,实现了策略接口或抽象类中定义的具体算法逻辑。

策略模式的工作流程如下:

  1. 客户端创建一个环境对象,并通过构造函数或设置方法将具体的策略对象传递给环境对象。

  2. 客户端根据需求选择合适的策略对象。

  3. 环境对象在执行任务时,会将具体的算法委派给当前持有的策略对象进行处理。

  4. 当需要切换算法时,客户端可以动态地替换环境对象的策略对象,实现不同的算法逻辑。

策略模式的优点包括灵活性、可替换性、可扩展性和代码复用性。它可以避免使用大量的条件语句或开关语句,将不同的算法逻辑封装在独立的策略类中,使系统更加灵活、易于扩展和维护。

关于本文

策略模式组成是通过环境,策略,具体实现三部分组成,本文将环境(context)通过注册器实现,实现自动化注入容器中,维护起来更方便。主要用于优化系统中多if判断场景,大一点的系统应该都能碰到,所以希望可以帮助到大家。

创建登陆策略


public interface LoginStrategy {

    /**
     * 检查参数
     * @return
     */
    default Result checkParam(LoginPatamDTO loginPatamDTO){
        return Result.err();
    }

    /**
     * 登陆
     * @param loginPatamDTO
     * @return
     */
    default Result<LoginUserInfoVO> login(LoginPatamDTO loginPatamDTO){
        return Result.err();
    }
}


创建环境(注册器)

将bean对象放在指定的map容器中。


@Component
public class LoginCommonStrategyRegistry {

    @Autowired
    private BeanFactory beanFactory;


    private Map<LoginTypeEnum, LoginStrategy> loginrCommonStrategyMap = new ConcurrentHashMap<>();

    public Map<LoginTypeEnum, LoginStrategy> mapGet() {
        return this.loginrCommonStrategyMap;
    }

    /**
     * 初始化策略
     */
    public void registry(LoginTypeEnum loginTypeEnum, Class glass) {
        LoginStrategy funderCommonStrategy = (LoginStrategy)beanFactory.getBean(glass);
        this.loginrCommonStrategyMap.put(loginTypeEnum, funderCommonStrategy);
    }
}

简化代码 创建父类对象,也方便后期扩展

@PostConstruct 是 Java 中的一个注解,用于标记一个方法,在对象创建之后(通过构造函数创建并注入依赖后),在依赖注入完成之后立即执行。它的作用是在对象初始化阶段执行一些必要的操作,例如初始化资源、建立连接、加载数据等。


@Service
public abstract class AbstractLoginService implements LoginStrategy {

    private final LoginCommonStrategyRegistry loginCommonStrategyRegistry;
    public AbstractLoginService(LoginCommonStrategyRegistry loginCommonStrategyRegistry){
        this.loginCommonStrategyRegistry = loginCommonStrategyRegistry;
    }

    /**
     * 注册当前类到策略map中
     */
    @PostConstruct
    protected abstract void initRegistry();
}



具体策略实现类

用户名密码登陆


@Service
@Slf4j
public class UserNameLoginService extends AbstractLoginService {


    public UserNameLoginService(LoginCommonStrategyRegistry loginCommonStrategyRegistry) {
        super(loginCommonStrategyRegistry);
    }

    @Override
    protected void initRegistry() {
        super.loginCommonStrategyRegistry.registry(LoginTypeEnum.USERNAME_TYPE, this.getClass());
    }

    @Override
    public Result checkParam(LoginPatamDTO loginPatamDTO) {
        if (ObjectUtils.isEmpty(loginPatamDTO.getUserName())) {
            return Result.err("用户名不能为空");
        }
        if (ObjectUtils.isEmpty(loginPatamDTO.getPassword())) {
            return Result.err("密码不能为空");
        }
        return Result.suc();
    }

    @Override
    public Result<LoginUserInfoVO> login(LoginPatamDTO loginPatamDTO) {
        log.info("这里是用户名和密码登陆");
        if (loginPatamDTO.getUserName().equals("admin") && loginPatamDTO.getPassword().equals("admin")) {
            // 创建默认对象 模拟登陆返回
            LoginUserInfoVO loginUserInfoVO = LoginUserInfoVO.loginUserInfoVOCreate(loginPatamDTO.getLoginTypeEnum());
            return Result.suc(loginUserInfoVO);
        } else {
            return Result.err("用户名或者密码不正确");
        }
    }
}

手机号登陆


@Service
@Slf4j
public class PhoneLoginService extends AbstractLoginService {


    public PhoneLoginService(LoginCommonStrategyRegistry loginCommonStrategyRegistry) {
        super(loginCommonStrategyRegistry);
    }

    @Override
    protected void initRegistry() {
        super.loginCommonStrategyRegistry.registry(LoginTypeEnum.PHONE_TYPE, this.getClass());
    }

    @Override
    public Result checkParam(LoginPatamDTO loginPatamDTO) {
        if (ObjectUtils.isEmpty(loginPatamDTO.getPhone())) {
            return Result.err("手机号不能为空");
        }
        if (ObjectUtils.isEmpty(loginPatamDTO.getSmsCode())) {
            return Result.err("验证码不能为空");
        }
        return Result.suc();
    }

    @Override
    public Result<LoginUserInfoVO> login(LoginPatamDTO loginPatamDTO) {
        log.info("这里是手机号登陆");
        if (loginPatamDTO.getPhone().equals("17777777777") && loginPatamDTO.getSmsCode().equals("7777")) {
            // 创建默认对象 模拟登陆返回
            LoginUserInfoVO loginUserInfoVO = LoginUserInfoVO.loginUserInfoVOCreate(loginPatamDTO.getLoginTypeEnum());
            return Result.suc(loginUserInfoVO);
        } else {
            return Result.err("验证码不正确");
        }
    }
}



触发点

如果没有写策略模式 那代码应该是这样

if(type.equals("用户名登陆")){
	// 用户名密码登陆验证
}else if(type.equals("手机号登陆")){
	// 手机号登陆
}else if(.......){
 ....
}

使用策略模式触发点确实简化了不少代码,让代码优雅起来了。后续扩展啥的都很方便。

@RestController
public class LoginController {
    /**
     * 模拟redis
     */
    public static Map<String, String> userInforedis = new ConcurrentHashMap<>();

    private final LoginCommonStrategyRegistry loginCommonStrategyRegistry;

    public LoginController(LoginCommonStrategyRegistry loginCommonStrategyRegistry) {
        this.loginCommonStrategyRegistry = loginCommonStrategyRegistry;
    }


    @RequestMapping(value = "/a/login")
    public Result<LoginUserInfoVO> login(LoginPatamDTO loginPatamDTO) {
        if (ObjectUtils.isEmpty(loginPatamDTO.getLoginTypeEnum())) {
            return Result.err("无效的登陆方式");
        }
        LoginStrategy loginStrategy = loginCommonStrategyRegistry.mapGet().get(
                loginPatamDTO.getLoginTypeEnum());
        if (loginStrategy == null) {
            return Result.err("无效的登陆方式");
        }
        Result result = loginStrategy.checkParam(loginPatamDTO);
        if (result.getCode() == 0) {
            return result;
        }
        Result<LoginUserInfoVO> login = loginStrategy.login(loginPatamDTO);
        if (login.getCode() == 1) {
            userInforedis.put(login.getDate().getToken(), JSON.toJSONString(login.getDate()));
        }
        return login;
    }

}

调试



http://127.0.0.1:8080/a/login?loginTypeEnum=USERNAME_TYPE&userName=admin&password=admin

{"code":1,"msg":"成功","date":{"token":"57a44abe-e731-40f1-a9f6-1069710ac6fa","userInfoBO":{"userId":"这个是id","userRealName":"这个是名字","userSex":"这个是性别","userDeptIds":"这个是部门","loginTypeEnum":"USERNAME_TYPE"}}}

简言

在接入策略模式前一定要想好调用交互逻辑,如果系统有现成的最好使用现成的。当然也要避免循环依赖注入。

希望本文可以帮到大家。

更多推荐

Guava精讲(三)-Caches,同步DB数据到缓存

在开发中,我们经常需要从数据库中读取数据并进行频繁的读取操作。缓存在各种场景中都有运用,例如,当一个值的计算或检索成本很高,而且在某个输入中需要多次使用该值时,就应该考虑使用缓存,因此将数据缓存在内存中可以显著提高应用程序的性能。问题描述假设我们正在开发一个电子商务网站,需要频繁地显示商品信息。商品信息存储在数据库中,

SpringMVC之JSON返回&异常处理机制

json处理统一异常处理1.json处理//pom.xml<?xmlversion="1.0"encoding="UTF-8"?><projectxmlns="http://maven.apache.org/POM/4.0.0"xmlns:xsi="http://www.w3.org/2001/XMLSchema-in

服务器搭建(TCP套接字)-select版(服务端)

一、select头文件#include<sys/select.h>二、select原型intselect(intnfds,fd_set*readfds,fd_set*writefds,fd_set*exceptfds,structtimeval*timeout);select()是一个系统调用函数,用于在多个文件描述符

K8S架构原理

目录一、k8s概述1、什么是k8s?2、特性3、主要功能二、集群架构与组件1.Master组件(1)Kube-apiserver(2)Kube-controller-manager(3)Kube-scheduler调度算法:2.配置存储中心3.Node组件(1)Kubelet(2)Kube-Proxy(3)docker

【SpringMVC】之自定义注解

文章目录一、Java注解1.1简介1.2分类1.2.1JDK基本注解1.2.2JDK元注解1.3自定义注解二、使用自定义注解2.1案例一(获取类与方法上的注解值)2.2案例二(获取类属性上的注解属性值)2.3案例三(获取参数修饰注解对应的属性值)三、Aop自定义注解的应用一、Java注解1.1简介Java注解是附加在代

[论文阅读]Coordinate Attention for Efficient Mobile Network Design

摘要最近关于移动网络设计的研究已经证明了通道注意力(例如,theSqueeze-and-Excitationattention)对于提高模型的性能有显著的效果,但它们通常忽略了位置信息,而位置信息对于生成空间选择性注意图非常重要。在本文中,我们提出了一种新的移动网络注意力机制,将位置信息嵌入到通道注意力中,我们称之为“

【论文笔记】Baidu Apollo EM Motion Planner

文章目录AbstractI.INTRODUCTIONA.MultilaneStrategyB.Path-SpeedIterativeAlgorithmC.DecisionsandTrafficRegulationsII.EMPLANNERFRAMEWORKWITHMULTILANESTRATEGYIII.EMPLANN

nginx知识点详解:反向代理+负载均衡+动静分离+高可用集群

一、nginx基本概念1.nginx是什么,做什么事情?Nginx是一个高性能的HTTP和反向代理服务器,特点是占有内存少,并发能力强。Nginx转为性能优化而开发,能经受高负载考验。支持热部署,启动容易,运行时间长。2.反向代理正向代理:在客户端(浏览器)配置代理服务器,通过代理服务器进行互联网访问。反向代理:客户端

Python 08学习之文件操作

😀前言欢迎来到Python08学习之文件操作。在本文中,我们将介绍计算机中常见的文本文件和二进制文件,并探讨在Python中操作文件的步骤和相关函数/方法。通过学习本文,您将能够了解如何使用Python打开、读取、写入和关闭文件,以及如何按行读取文件内容。希望您能够通过本文提高您的Python文件操作能力,并且在实际

华为HCIA(二)

今天是第二天(一题一笔记)FTP(文件传输协议)使用的端口号是20和21控制层面用的是21DHCP(IP地址和子网掩码)服务器分配IP地址默认的租期是24小时Tnlnet协议(网络层)默认使用的服务器端口号是23完成链路认证后,STA要通过Association发起链路服务协商WLAN通过SSID来区分不同的网络基于M

【PickerView案例10-国旗选择界面02 Objective-C预言】

一、好了,我们继续来实现这个国旗选择界面:1.它的界面里面,是不是很简单,就一个UIPickerView,就完事儿了然后,显示的每一行内容呢,1)一个文字Label2)一个图片那大家应该有意识,它返回的应该是一个View,对吧,代理方法里面,有一个返回View的,viewForRow:viewForRowInCompo

热文推荐