【个人笔记】用户注册登录思路及实现 springboot+mybatis+redis

news2025/4/3 4:40:11

基本思路

获取验证码接口

验证码操作用了com.pig4cloud.plugin的captcha-core这个库。在这里插入图片描述

  1. AccountControl的"/checkCode"接口代码,通过ArithmeticCaptcha生成一张验证码图片,通过text()函数得到验证码的答案保存到变量code,然后把图片转为base64方便传输保存到变量checkCodeBase64
  2. code作为参数传入函数saveCheckCode,把函数返回结果保存到变量checkCodeKey
  3. 创建一个Map,checkCodeBase64checkCodeKey放进去,然后丢给getSuccessResponseVO()当作响应消息返回。
@RequestMapping("/checkCode")
	public ResponseVO checkCode(){
		//生成验证码
		ArithmeticCaptcha captcha = new ArithmeticCaptcha(100,42);
		//获取验证码
		String code = captcha.text();
		//生成base64
		String checkCodeBase64 = captcha.toBase64();
		//把验证码存入redis
		String checkCodeKey = redisComponet.saveCheckCode(code);
		Map<String,String> result = new HashMap<>();
		result.put("checkCode",checkCodeBase64);
		result.put("checkCodeKey",checkCodeKey);
		return getSuccessResponseVO(result);
	}
  1. saveCheckCode函数负责把验证码答案code保存到redis,生成一个随机的 UUID作为存到redis的key,把这个key返回出去。
public String saveCheckCode(String code){
    String checkCodeKey = UUID.randomUUID().toString();
    redisUtils.setex(Constants.REDIS_KEY_CHECK_CODE+checkCodeKey,code,Constants.REDIS_KEY_EXPIRE_ONE_MIN);
    return checkCodeKey;
}

接口执行结果:
在这里插入图片描述

用户注册

在这里插入图片描述

  1. 根据传入的验证码答案checkCode判断是否与redis储存的值相等。
  2. 验证码通过后执行register服务函数。
  3. 执行完try的代码,执行finally代码删除redis相应的键值。
@RequestMapping("/register")
public ResponseVO register(@NotEmpty @Email @Size(max = 150) String email,
						   @NotEmpty @Size(max = 20) String nickName,
						   @NotEmpty @Pattern(regexp = Constants.REGEX_PASSWORD) String registerPassword,
						   @NotEmpty String checkCodeKey,
						   @NotEmpty String checkCode){
	try {
		if(!checkCode.equalsIgnoreCase(redisComponet.getCheckeCode(checkCodeKey))){
			throw new BusinessException("图片验证码不正确");
		}
		userInfoService.register(email,nickName,registerPassword);
		return getSuccessResponseVO(null);
	} catch (BusinessException e) {
           throw new RuntimeException(e);
       } finally {
		redisComponet.cleanCheckCode(checkCodeKey);
	}
}
  1. 检查邮箱和昵称是否已存在。
  2. mybatis操作数据库创建新注册的用户信息。
@Override
public void register(String email, String nickName, String registerPassword) throws BusinessException {
	UserInfo userInfo = this.userInfoMapper.selectByEmail(email);
	if(null!=userInfo){
		throw new BusinessException("邮箱已被注册");
	}
	UserInfo nickNameUser = this.userInfoMapper.selectByNickName(nickName);
	if(null!=nickNameUser){
		throw new BusinessException("昵称已被注册");
	}
	userInfo = new UserInfo();
	String userId = StringTools.getRandomNumber(Constants.USERID_LENGTH);
	userInfo.setUserId(userId);
	userInfo.setEmail(email);
	userInfo.setNickName(nickName);
	userInfo.setPassword(StringTools.encodeByMd5(registerPassword));
	userInfo.setJoinTime(new Date());
	userInfo.setStatus(UserStatusEnum.NORMAL.getCode());
	userInfo.setTotalCoin(0);
	this.userInfoMapper.insert(userInfo);
   }

执行结果:
输入对应的参数
注册
注册成功
在这里插入图片描述
检查数据库
加粗样式

登录接口

在这里插入图片描述

  1. 传入email、password、checkCodeKey和checkCode,校验验证码。
  2. 获取登录的ip地址,向登录服务函数login传入email、password和ip返回dto对象,把该dto对象保存。
  3. SaveTokenToCookie函数作用是保存dto里的token到浏览器中,方便前端获取。
  4. 执行完try的代码,执行finally代码删除redis相应的键值。
@RequestMapping("/login")
public ResponseVO login(HttpServletResponse response,
						@NotEmpty @Email String email,
						@NotEmpty String password,
						@NotEmpty String checkCodeKey,
						@NotEmpty String checkCode){
	try {
		if(!checkCode.equalsIgnoreCase(redisComponet.getCheckeCode(checkCodeKey))){
			throw new BusinessException("图片验证码不正确");
		}
        String ip = getIpAddr();
		TokenUserInfoDTO tokenUserInfoDTO = userInfoService.login(email	,password,ip);
		SaveTokenToCookie(response,tokenUserInfoDTO.getToken());
		return getSuccessResponseVO(tokenUserInfoDTO);
	} catch (BusinessException e) {
		throw new RuntimeException(e);
	} finally {
		redisComponet.cleanCheckCode(checkCodeKey);
	}
}

获取用户IP地址的函数

protected String getIpAddr() {
	HttpServletRequest request = ((ServletRequestAttributes) RequestContextHolder.getRequestAttributes()).getRequest();
	String ip = request.getHeader("x-forwarded-for");
	if (ip != null && ip.length() != 0 && !"unknown".equalsIgnoreCase(ip)) {
	    // 多次反向代理后会有多个ip值,第一个ip才是真实ip
	    if (ip.indexOf(",") != -1) {
	        ip = ip.split(",")[0];
	    }
	}
	if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {
	    ip = request.getHeader("Proxy-Client-IP");
	}
	if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {
	    ip = request.getHeader("WL-Proxy-Client-IP");
	}
	if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {
	    ip = request.getHeader("HTTP_CLIENT_IP");
	}
	if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {
	    ip = request.getHeader("HTTP_X_FORWARDED_FOR");
	}
	if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {
	    ip = request.getHeader("X-Real-IP");
	}
	if (ip == null || ip.length() == 0 || "unknown".equalsIgnoreCase(ip)) {
	    ip = request.getRemoteAddr();
	}
	return ip;
}

在浏览器保存token

protected void SaveTokenToCookie(HttpServletResponse response,String token) {
    Cookie cookie = new Cookie(Constants.TOKEN_WEB,token);
    cookie.setMaxAge(Constants.REDIS_KEY_EXPIRE_ONE_DAY / 1000 * 7);
    cookie.setPath("/");
    response.addCookie(cookie);
}

在这里插入图片描述

  1. 检验邮箱、密码和用户状态,抛出对应的异常。
  2. 登录信息无误,根据用户ID更新ip和最新登录时间。
  3. 对象拷贝,将一个对象userInfo的属性复制到DTO对象中(不会复制DTO中没有的属性)。
  4. 把DTO对象保存到redis中,token的内容是保存到redis的key值。
@Override
public TokenUserInfoDTO login(String email, String password, String ip) throws BusinessException {
	UserInfo userInfo = this.userInfoMapper.selectByEmail(email);
	if(null==userInfo || !userInfo.getPassword().equals(StringTools.encodeByMd5(password))){
		throw new BusinessException("用户名或密码错误");
	}
	if(UserStatusEnum.FORBIDDEN.getCode().equals(userInfo.getStatus())){
		throw new BusinessException("用户已被禁用");
	}
	UserInfo updateInfo = new UserInfo();
	updateInfo.setLastLogin(new Date());
	updateInfo.setLastIp(ip);
	this.userInfoMapper.updateByUserId(updateInfo,userInfo.getUserId());
	TokenUserInfoDTO tokenUserInfoDTO = CopyTools.copy(userInfo, TokenUserInfoDTO.class);
	redisComponet.saveTokenInfo(tokenUserInfoDTO);
	return tokenUserInfoDTO;
}

执行结果:
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

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

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

相关文章

聚类(Clustering)基础知识3

文章目录 一、聚类的性能评价1、聚类性能评价&#xff08;1&#xff09;聚类性能评价方法&#xff1a; 2、参考模型 (reference model)&#xff08;1&#xff09;数据集&#xff1a;&#xff08;2&#xff09;聚类结果&#xff1a;&#xff08;3&#xff09;参考模型&#xff1…

RK3588使用笔记:设置程序/服务开机自启

一、前言 一般将系统用作嵌入式设备时肯定要布置某些程序&#xff0c;这时候就需要对程序设置开机自己&#xff0c;否则每次都要人为启动&#xff0c;当有些嵌入式系统未连接显示屏或者无桌面环境去操作启动程序时&#xff0c;程序自启就是必须的了&#xff0c;本文介绍在纯li…

python实现股票数据可视化

最近在做一个涉及到股票数据清洗及预测的项目&#xff0c;项目中需要用到可视化股票数据这一功能&#xff0c;这里我与大家分享一下股票数据可视化的一些基本方法。 股票数据获取 目前&#xff0c;我已知的使用python来获取股票数据方式有以下三种: 爬虫获取&#xff0c;实现…

JavaScript DOM与元素操作

目录 DOM 树、DOM 对象、元素操作 一、DOM 树与 DOM 对象 二、获取 DOM 元素 1. 基础方法 2. 现代方法&#xff08;ES6&#xff09; 三、修改元素内容 四、修改元素常见属性 1. 标准属性 2. 通用方法 五、通过 style 修改样式 六、通过类名修改样式 1. className 属…

ARM向量表

向量表作用说明RVBAR在 AArch64 中&#xff0c;重置向量不再是异常向量表的一部分。 有复位向量的专用配置输入引脚和寄存器。在 AArch64 中&#xff0c;处理器从 IMPLEMENTAION‑DEFINED 地址开始执行&#xff0c; 该地址由硬件输入引 脚RVBARADDR定义&#xff0c; 可以通过 R…

leetcode刷题日记——除自身以外数组的乘积

[ 题目描述 ]&#xff1a; [ 思路 ]&#xff1a; 题目要求获取数组中每个元素除自己以外的各元素的乘积最简单的方法就是算出数组所有元素的乘积&#xff0c;然后除以自身&#xff0c;即可得到除自身外各元素的乘积 但要考虑到其自身为0的情况&#xff0c;即当期自身为0时&am…

【信奥一本通提高篇】基础算法之贪心算法

原文 https://bbs.fmcraft.top/blog/index.php/archives/22/ 贪心算法 概述 近年来的信息学竞赛试题&#xff0c;经常出现求一个问题的可行解或最优解的题目。这类问题就是我们通常所说的最优化问题。贪心算法是求解这类问题的一种常用算法。在众多的算法中&#xff0c;贪心…

PyQt6实例_批量下载pdf工具_批量pdf网址获取

目录 前置&#xff1a; 步骤&#xff1a; step one 安装包 step two 获取股票代码 step three 敲代码&#xff0c;实现 step four 网址转pdf网址 视频 前置&#xff1a; 1 本系列将以 “PyQt6实例_批量下载pdf工具”开头&#xff0c;放在 【PyQt6实例】 专栏 2 本节讲…

KMeans算法案例

KMeans算法案例 案例介绍 已知&#xff1a;客户性别、年龄、年收入、消费指数 需求&#xff1a;对客户进行分析&#xff0c;找到业务突破口&#xff0c;寻找黄金客户 数据集共包含顾客的数据, 数据共有 4 个特征, 数据共有 200 条。接下来&#xff0c;使用聚类算法对具有相似…

IDApro直接 debug STM32 MCU

使用IDA pro 逆向分析muc 固件的时候&#xff0c; 难免要进行一些动态的debug&#xff0c;来进一步搞清楚一些内存的数据、算法等&#xff0c;这时候使用远程debug 的方式直接在mcu上进行debug 最合适不过了。 不过有个前提条件就是一般来说有的mcu 会被运行中的代码屏蔽 RDP、…

六十天前端强化训练之第三十六天之E2E测试(Cypress)大师级完整指南

欢迎来到编程星辰海的博客讲解 看完可以给一个免费的三连吗&#xff0c;谢谢大佬&#xff01; 目录 一、知识讲解 1. E2E测试核心概念 2. Cypress框架特性 3. 工作原理 4. 测试金字塔定位 二、核心代码示例&#xff1a;用户登录全流程测试 三、实现效果展示 四、学习要…

20250330-傅里叶级数专题之离散傅里叶变换(5/6)

5. 傅里叶级数专题之离散傅里叶变换 推荐视频: 工科生以最快的速度理解离散傅立叶变换(DFT) 哔哩哔哩 20250328-傅里叶级数专题之数学基础(0/6)-CSDN博客20250330-傅里叶级数专题之傅里叶级数(1/6)-CSDN博客20250330-傅里叶级数专题之傅里叶变换(2/6)-CSDN博客20250330-傅里叶…

3.29:数据结构-绪论线性表-上

一、时间复杂度 1、ADT 2、定义法计算时间复杂度&#xff1a;统计核心语句的总执行次数 &#xff08;1&#xff09;例题1&#xff0c;与2022年的真题对比着写 此题关键在于求和公式的转化&#xff0c;类型为&#xff1a;线性循环嵌套非线性循环 2022年那道题如果考场上实在脑…

大模型架构记录13【hr agent】

一 Function calling 函数调用 from dotenv import load_dotenv, find_dotenvload_dotenv(find_dotenv())from openai import OpenAI import jsonclient OpenAI()# Example dummy function hard coded to return the same weather # In production, this could be your back…

conda 清除 tarballs 减少磁盘占用 、 conda rename 重命名环境、conda create -n qwen --clone 当前环境

&#x1f947; 版权: 本文由【墨理学AI】原创首发、各位读者大大、敬请查阅、感谢三连 &#x1f389; 声明: 作为全网 AI 领域 干货最多的博主之一&#xff0c;❤️ 不负光阴不负卿 ❤️ 文章目录 conda clean --tarballsconda rename 重命名环境conda create -n qwen --clone …

pycharm相对路径引用方法

用于打字不方便&#xff0c;以下直接手写放图&#xff0c;直观理解

新能源智慧灯杆的智能照明系统如何实现节能?

叁仟新能源智慧灯杆的智能照明系统可通过以下多种方式实现节能&#xff1a; 智能调光控制 光传感器技术&#xff1a;在灯杆上安装光传感器&#xff0c;实时监测周围环境的光照强度。当环境光线充足时&#xff0c;如白天或有其他强光源时&#xff0c;智能照明系统会自动降低路…

Jenkins教程(自动化部署)

Jenkins教程(自动化部署) 1. Jenkins是什么&#xff1f; Jenkins是一个开源的、提供友好操作界面的持续集成(CI)工具&#xff0c;广泛用于项目开发&#xff0c;具有自动化构建、测试和部署等功能。Jenkins用Java语言编写&#xff0c;可在Tomcat等流行的servlet容器中运行&…

行业智能体大爆发,分布式智能云有解

Manus的一夜爆红&#xff0c;在全球范围内引爆关于AI智能体的讨论。 与过去一般的AI助手不同&#xff0c;智能体&#xff08;AI Agent&#xff09;并非只是被动响应&#xff0c;而是主动感知、决策并执行的应用。Gartner预测&#xff0c;到2028年&#xff0c;15%的日常工作决策…

日语Learn,英语再认识(5)

This is a dedicated function — it exists solely to solve this case. This is a dedicated function. It’s a dedicated method for solving this case. 其他备选词&#xff08;但没dedicated精准&#xff09;&#xff1a; special → 含糊&#xff0c;有时只是“特别”…