实际项目开发:Spring集成Redis,并实现短信登录功能

2024-06-23 23:04

本文主要是介绍实际项目开发:Spring集成Redis,并实现短信登录功能,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

redis新手,学了几种基本数据类型,却不知道怎么使用?
总是一边学一边忘?
学会了Redis的大多数使用命令,却不知道如何在项目中使用?
本文将从实际出发,为大家解决这些问题。
我是蚊子码农,欢迎各位的点赞、关注和收藏,有了你们的激励,我会带来更好的作品。

一、环境准备

从环境出发。

第一,基础环境:
  1. JDK版本:JDK1.8。
  2. 构建工具:Maven。
  3. 开发工具:IDEA。
  4. 基础依赖:SpringBoot通用依赖。
  5. 前后端验证机制:JWT令牌机制【如果是Token,可以看成一样;如果是cookie或session,则需要按情况改变】
第二,个人环境:
  1. Redis环境(操作系统):Windows10
  2. 使用的Redis客户端:lettuce

二、在pom文件导入Redis的相关依赖、配置Redis相关信息

注意:本文的基础,是已经安装好Redis,并且学会启动Redis服务器的用户。
简要安装步骤:来到Redis官网,按照版本下载—》解压Redis的压缩包到某个目录—》在这个解压包里,找到redis-server.exe打开服务器。【这个方法的好处是,不用配置系统变量,当然,也不能随时随地通过CMD命令启动Redis】

        <!--    redis的依赖    --><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-data-redis</artifactId></dependency><dependency><groupId>org.apache.commons</groupId><artifactId>commons-pool2</artifactId><version>2.11.1</version>

完成依赖的导入后,需要配置appliacation.yml文件的配置信息。
这个信息和MySQL非常类似,如下:

  redis:host: 127.0.0.1 # 服务器的IP地址port: 6379 # 默认端口号# 默认密码是null,如果设定密码则修改password: # lettuce客户端lettuce:# 连接池的配置信息,和线程池类似pool:max-active: 8   # 最大连接数,负数表示无限制max-wait: -1    # 最大阻塞等待时间,负数则表示无限制max-idle: 8     # 连接池最大空闲连接

三、写Redis操作的通用类RedisUtil【基于StringRedisTemplate类】

在Spring集成Redis的项目中,通常可以使用RedisTemplate和StringRedisTemplate来操作Redis。
RedisTemplate<K, V>,是一个泛型类,允许很多种结构,灵活性很高。
而StringRedisTemplate<String, String>,继承自RedisTemplate,只允许字符串对字符串的映射,相对来说灵活性较差,但是开发字符串结构的业务,非常方便高效。
本文的短信验证机制,只需要String对String即可,因此,使用StringRedisTemplate开发。
工具类代码如下,本质只有读、写两个操作:

// 可以写接口和实现类,来充分解耦,本文不解耦
@Component
public class RedisUtil {@Autowiredprivate StringRedisTemplate stringRedisTemplate;// 向Redis中,设定验证码【手机号映射code的方式】public void redisSetCode(String phone, String code){// 设定手机号、验证码、5分钟的过期时间【最后的参数,是时间单位】stringRedisTemplate.opsForValue().set(phone, code, 60 * 5, TimeUnit.SECONDS);}// 从Redis中得到验证码,如果返回null,表示没有这个phone,在此不做验证public String redisGetCode(String phone){return stringRedisTemplate.opsForValue().get(phone);}
}

四、短信登陆功能的开发步骤

短信登录验证是市面上最火的登录验证方式之一。
从B/S的交互角度来看,它一共有4步:

第一,用户输入手机号,并向服务器发送请求。
第二,服务器验证手机号有效性,生成验证码,并将验证码返回客户端。
第三,用户输入手机号、验证码,并向服务器发送请求。
第四,服务器验证二者的对应关系和有效性,返回结果。

我们可以发现,这个验证机制的关键只有2个,手机号、验证码。
二者有一对一的关系,为了保证安全,验证码还必须有过期策略。因此,在没有其它业务逻辑的情况下,可以使用redis方便地完成这个功能。
不使用MySQL的原因:该验证机制较为简单,使用MySQL却需要建立一张难以变化的表,很不值得,加上访问速度也没有Redis快。
本文面向后端用户,所以前端的开发流程在此不说明,我们重点关注以下步骤:

第一,得到手机号。

这一步,需要做至少3件事:

  1. 手机号有效性检测【判断数据库有没有这个人(如果该验证机制,不允许使用手机号注册的话)】
  2. 生成验证码
  3. 将验证码写进Redis
  4. 注:借用短信API将验证码发给用户。【这一步不好写】
第二,得到手机号、验证码

需要做2件事:

  1. 判断手机号与验证码是否对应
  2. 如果对应,返回个人信息;如果不对应,报错。

五、后端得到手机号

在开发前,我们要保证,前端传来的手机号没有异常。【在本文不判断手机号null的情况】

第一步,搜索数据库,查找有没有这个人

说明:这一步可以省略,可以直接返回验证码,等到用户拿到验证码后,再从数据库拿数据判断,这样可以节省一点后端资源。【不过,用户使用起来是比较难受的,所以建议先判断】
判断的代码比较简单,在此不说明。

第二步,生成验证码

这个业务,可能需要各种复用,干脆写成一个工具类。
代码如下:

public class GeneCodeUtil {// 验证码的范围,如果想使用复杂字符,可以写i、g、a、b等public static final String VERIFY_CODES = "1234567890";// 验证码生成函数,指定验证码的位数【比如4位验证码:1234、5144】public static String CodeGenerate(int verifySize){String sources = VERIFY_CODES;int codesLen = sources.length();// 以当前的毫秒数为种子,生成随机数Random rand = new Random(System.currentTimeMillis());// 使用StringBuilder结构存储数据StringBuilder sb = new StringBuilder(verifySize);for(int i = 0; i < verifySize; i++){// 添加到sb里sb.append(sources.charAt(rand.nextInt(codesLen-1)));}return sb.toString();}
}
第三,将手机号、验证码写入Redis数据库。

由于我们已经写好了Redis的工具类,在此调用工具类的set方法即可。
下面的代码,是手机号发送的业务逻辑:
下面的一些类,比较简单,就不分开讲解了,我都在第一次使用时,做了说明

	// 验证有效性,生成验证码,写入Redis数据库@PostMapping("/login")public CodeDTO loginGenerateCode(@RequestBody PhoneCodeDTO phoneCodeDTO){// CodeDTO类:包含通用的返回信息、返回状态码;比如【1,验证码成功发送】// PhoneCodeDTO类:包含验证码、手机号,为了方便接收前端的数据,现在验证码属性为nullString phone = phoneCodeDTO.getPhone();        // 查找有没有这个人String tempPhone = new String(phone);// Public类:一个数据类,不重要LambdaQueryWrapper<Public> publicLambdaQueryWrapper = new LambdaQueryWrapper<>();publicLambdaQueryWrapper.eq(Public::getPubNum,Long.parseLong(tempPhone));// 得到这个人Public pub = publicService.getOne(publicLambdaQueryWrapper);// 空表示没有,说明失败if(pub == null){// codeFail:返回CodeDTO的对象,输入的数据,即为返回信息return CodeDTO.codeFail("不存在此用户");}// codeRedisAndSend:send函数做了2件事。。1.生成验证码;2.写入Redis服务器codeRedisAndSend.send(phone);// 可以在这里,写一些代码,调用信息发送API,将信息发送给用户// codeSuccess:返回CodeDTO对象,输入的数据为返回信息return CodeDTO.codeSuccess("验证码发送成功");}

六、阶段效果演示

使用Postman,发送请求给服务器
不存在用户时:
用户不存在时的展示
将该用户添加到数据库表中后:
验证码成功发布

七、后端拿到验证码、手机号

其实业务逻辑比较简单,所以直接看代码即可。

 	// 请求URL@RequestMapping("/codeLogin")// 效验验证码,如果有这个人,返回这个人的数据,JWT令牌public PublicDataDTO loginByCode(@RequestBody PhoneCodeDTO phoneCodeDTO){// PublicDataDTO类:包含用户信息、JWT令牌及一些附带信息// PhoneCodeDTO类:包含验证码、手机号信息String phone = phoneCodeDTO.getPhone();// 判断手机号、验证码是否对应,getCode方法是从Redis服务器拿到验证码if( phoneCodeDTO.getCode().equals(codeRedisAndSend.getCode(phone)) ){LambdaQueryWrapper<Public> publicLambdaQueryWrapper = new LambdaQueryWrapper<>();publicLambdaQueryWrapper.eq(Public::getPubNum,Long.parseLong(phone));// 得到此人信息Public pub = publicService.getOne(publicLambdaQueryWrapper);// getPublicDataDTO方法:返回一个PublicDataDTO对象// JWT令牌是JWT工具类生成的,传给前端,用作后续效验return PublicDataDTO.getPublicDataDTO(pub);}// 不相同,则返回nullreturn PublicDataDTO.getPublicDataDTO(null);}

八、效果演示

同样使用Postman作为演示,假设我们知道验证码是1666
那么,输入错误的验证码:
错误验证码
我们输入正确的验证码
正确验证码

九、结语

我是蚊子码农,如有补充或者疑问,欢迎在评论区留言。个人的知识体系可能没有那么完善,希望各位多多指正,谢谢大家。

这篇关于实际项目开发:Spring集成Redis,并实现短信登录功能的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1088529

相关文章

Ubuntu 24.04启用root图形登录的操作流程

《Ubuntu24.04启用root图形登录的操作流程》Ubuntu默认禁用root账户的图形与SSH登录,这是为了安全,但在某些场景你可能需要直接用root登录GNOME桌面,本文以Ubuntu2... 目录一、前言二、准备工作三、设置 root 密码四、启用图形界面 root 登录1. 修改 GDM 配

Spring Boot中的路径变量示例详解

《SpringBoot中的路径变量示例详解》SpringBoot中PathVariable通过@PathVariable注解实现URL参数与方法参数绑定,支持多参数接收、类型转换、可选参数、默认值及... 目录一. 基本用法与参数映射1.路径定义2.参数绑定&nhttp://www.chinasem.cnbs

Redis中Stream详解及应用小结

《Redis中Stream详解及应用小结》RedisStreams是Redis5.0引入的新功能,提供了一种类似于传统消息队列的机制,但具有更高的灵活性和可扩展性,本文给大家介绍Redis中Strea... 目录1. Redis Stream 概述2. Redis Stream 的基本操作2.1. XADD

JAVA中安装多个JDK的方法

《JAVA中安装多个JDK的方法》文章介绍了在Windows系统上安装多个JDK版本的方法,包括下载、安装路径修改、环境变量配置(JAVA_HOME和Path),并说明如何通过调整JAVA_HOME在... 首先去oracle官网下载好两个版本不同的jdk(需要登录Oracle账号,没有可以免费注册)下载完

Spring StateMachine实现状态机使用示例详解

《SpringStateMachine实现状态机使用示例详解》本文介绍SpringStateMachine实现状态机的步骤,包括依赖导入、枚举定义、状态转移规则配置、上下文管理及服务调用示例,重点解... 目录什么是状态机使用示例什么是状态机状态机是计算机科学中的​​核心建模工具​​,用于描述对象在其生命

Spring Boot 结合 WxJava 实现文章上传微信公众号草稿箱与群发

《SpringBoot结合WxJava实现文章上传微信公众号草稿箱与群发》本文将详细介绍如何使用SpringBoot框架结合WxJava开发工具包,实现文章上传到微信公众号草稿箱以及群发功能,... 目录一、项目环境准备1.1 开发环境1.2 微信公众号准备二、Spring Boot 项目搭建2.1 创建

Java中Integer128陷阱

《Java中Integer128陷阱》本文主要介绍了Java中Integer与int的区别及装箱拆箱机制,重点指出-128至127范围内的Integer值会复用缓存对象,导致==比较结果为true,下... 目录一、Integer和int的联系1.1 Integer和int的区别1.2 Integer和in

SpringSecurity整合redission序列化问题小结(最新整理)

《SpringSecurity整合redission序列化问题小结(最新整理)》文章详解SpringSecurity整合Redisson时的序列化问题,指出需排除官方Jackson依赖,通过自定义反序... 目录1. 前言2. Redission配置2.1 RedissonProperties2.2 Red

IntelliJ IDEA2025创建SpringBoot项目的实现步骤

《IntelliJIDEA2025创建SpringBoot项目的实现步骤》本文主要介绍了IntelliJIDEA2025创建SpringBoot项目的实现步骤,文中通过示例代码介绍的非常详细,对大家... 目录一、创建 Spring Boot 项目1. 新建项目2. 基础配置3. 选择依赖4. 生成项目5.

nginx 负载均衡配置及如何解决重复登录问题

《nginx负载均衡配置及如何解决重复登录问题》文章详解Nginx源码安装与Docker部署,介绍四层/七层代理区别及负载均衡策略,通过ip_hash解决重复登录问题,对nginx负载均衡配置及如何... 目录一:源码安装:1.配置编译参数2.编译3.编译安装 二,四层代理和七层代理区别1.二者混合使用举例