在Spring项目中使用Maven和BCrypt来实现修改密码功能
简介
在数字时代,信息安全的重要性不言而喻,尤其当涉及到个人隐私和账户安全时。每天,无数的用户登录各种在线服务,从社交媒体到银行账户,再到电子邮件和云存储服务。这些服务的背后,是复杂的系统架构,其中包含着用户最为敏感的数据——密码。
过去,简单的加密方法和弱密码策略导致了许多严重的数据泄露事件。例如,2013年雅虎(Yahoo)遭遇的大规模数据泄露事件,影响了数十亿的用户账户,部分原因就是由于使用了不够安全的密码存储技术。再如2016年的LinkedIn数据泄露事件,尽管该公司使用了SHA-1散列算法对密码进行了处理,但未加盐的密码散列最终还是被破解,暴露了用户的隐私。
这些事件引发了行业对于密码安全的深刻反思,促使开发者和安全专家寻找更安全的解决方案。BCrypt作为一种适应性强且经过时间考验的密码哈希算法,成为了现代密码安全的基石。它不仅能够有效抵御暴力破解和彩虹表攻击,还能通过增加工作因子来适应未来计算能力的增长。
在本文中,我们将深入探讨如何在Spring项目中利用Maven和BCrypt来实现一个安全的密码修改功能。这不仅仅是关于代码实现的问题,更是一次对密码安全重要性的重申,以及对如何在实际应用中践行这一原则的示范。我们将从环境搭建开始,逐步构建出一个既实用又安全的密码修改流程,确保即使在最恶劣的情况下,用户的密码也能得到妥善保护。

通过本文的学习,你将获得宝贵的实践经验,了解如何在自己的项目中实施类似的解决方案,从而提升应用的安全性,给用户提供更加安心的在线体验。在接下来的内容中,我们将一步步解析实现过程,从添加依赖到编写核心业务逻辑,直至完成完整的功能测试,确保每一步都遵循最佳的安全实践。
controller(UserController)
/*** 修改密码*/@PutMapping("/update/pwd/{id}")public Result update(@PathVariable("id") long id, @RequestBody ChangePasswordVo changePasswordVo) {try{userService.changePassword(changePasswordVo,id);return Result.success("修改成功!");}catch (Exception e){// 捕获异常,获取异常信息String message = e.getMessage();// 如果修改失败,返回失败的结果,并附带异常信息return Result.failed(message);}}
-
注解
@PutMapping("/update/pwd/{id}")@PutMapping是Spring MVC中的一个注解,用于处理HTTP PUT请求。"/update/pwd/{id}"是该方法的URL路径。其中{id}是一个路径变量,用于接收用户ID。
-
方法定义
public Result update(...)- 这是一个公开的方法,名为
update,返回一个Result对象。 Result很可能是一个自定义的响应类,通常用于封装API的响应结果,包括状态码、消息和数据等。
- 这是一个公开的方法,名为
-
方法参数
@PathVariable("id") long id:这是从URL路径中提取的id变量。@PathVariable注解告诉Spring MVC从URL中提取名为id的变量值,并将其转换为long类型。@RequestBody ChangePasswordVo changePasswordVo:@RequestBody注解用于将HTTP请求体中的JSON数据转换为ChangePasswordVo类型的对象。ChangePasswordVo可能是一个包含旧密码和新密码等信息的DTO(数据传输对象)。
-
方法体
- 首先,它尝试调用
userService.changePassword(changePasswordVo,id);。这里假设userService是一个已经注入的服务类,用于处理与用户相关的业务逻辑。changePassword方法可能会根据提供的用户ID和密码信息来更新用户的密码。 - 如果上述操作成功,方法将返回一个表示成功的
Result对象,并附带消息“修改成功!”。 - 如果在更新密码的过程中发生异常(如数据库错误、密码验证失败等),
catch块将捕获该异常,并获取其消息。然后,它返回一个表示失败的Result对象,并附带异常的消息。
- 首先,它尝试调用
entity(VO-ChangePasswordVo)
@Data
public class ChangePasswordVo implements Serializable {/*** 旧的密码*/private String oldpassword;/*** 新的密码*/private String newpassword;}
-
注解
@Data: 这是一个Lombok库提供的注解。当你添加@Data到类上时,Lombok会自动为这个类生成getter、setter、equals、hashCode和toString方法。这可以大大减少模板代码的数量,使代码更加简洁。implements Serializable: 这表示该类实现了Serializable接口。Serializable是一个标记接口,用于指示一个类的对象可以被序列化。序列化是将对象状态转换为字节流,以便可以将其写入文件或发送到网络上的另一个位置。如果该类或其成员类(如果它们不是基本类型或String、数组等)需要被序列化,则必须实现这个接口。
-
成员变量
private String oldpassword;: 这是一个私有字符串类型的成员变量,用于存储旧的密码。但是,从Java的命名约定来看,变量名应该使用驼峰命名法(camelCase),并且首字母小写。因此,更合适的命名可能是oldPassword。private String newpassword;: 同样,这是一个私有字符串类型的成员变量,用于存储新的密码。按照Java的命名约定,它应该被命名为newPassword。
-
注释
- 每个成员变量上方都有注释,描述了该变量的用途。这是一个很好的做法,因为它增加了代码的可读性。但是,请注意,这些注释是用中文写的,而在国际项目中,通常建议使用英文注释。
Service
UserService
void changePassword(ChangePasswordVo changePasswordVo, Long id);
-
返回类型 (
void): 方法前面有一个void关键字,表示这个方法没有返回值。也就是说,当你调用这个方法时,它不会返回任何值或对象。 -
方法名 (
changePassword): 这是方法的名称,即changePassword。当你想在代码的其他部分调用这个方法时,你会使用这个名字。 -
参数:
-
ChangePasswordVo changePasswordVo: 这是方法的第一个参数。ChangePasswordVo: 这是参数的类型。它可能是一个数据传输对象(DTO),用于封装与更改密码相关的数据,如旧密码、新密码等。changePasswordVo: 这是参数的名称。在方法内部,你可以使用这个名称来引用传入的ChangePasswordVo对象。
-
Long id: 这是方法的第二个参数。Long: 这是参数的类型,表示这是一个长整型(64位整数)数据。id: 这是参数的名称。在方法内部,你可以使用这个名称来引用传入的id值。从参数名可以推测,这个id可能表示用户的唯一标识符(如用户ID)。
-
UserServiceImpl
@Overridepublic void changePassword(ChangePasswordVo changePasswordVo, Long id){// 根据id查询用户信息User user = userMapper.selectById(id);// 判断原密码是否正确if(!BCrypt.checkpw(changePasswordVo.getOldpassword(),user.getPassword())){throw new RuntimeException("原密码不正确");}// 设置新密码user.setPassword(BCrypt.hashpw(changePasswordVo.getNewpassword(), BCrypt.gensalt()));// 调用Mapper的updateById方法更新用户信息userMapper.updateById(user);}
-
@Override: 这是一个Java注解,它告诉编译器这个方法是从超类或接口中继承或实现的。使用
@Override注解可以确保你正确地重写了父类或接口中的方法,如果没有正确重写(例如方法签名不匹配),编译器会报错。 -
public void changePassword(ChangePasswordVo changePasswordVo, Long id): 这是方法的声明部分。
public:表示这是一个公共方法,可以从任何其他类中被访问。void:表示该方法没有返回值。changePassword:是方法的名称。ChangePasswordVo changePasswordVo和Long id:是方法的参数。ChangePasswordVo可能是一个数据传输对象(DTO),用于封装密码更改请求所需的信息(如旧密码和新密码)。id则是用户的唯一标识符。
-
User user = userMapper.selectById(id);:
- 使用
userMapper(可能是MyBatis的Mapper或类似的ORM工具)的selectById方法根据提供的id从数据库中查询用户信息,并将查询到的用户信息存储在user变量中。
- 使用
-
if(!BCrypt.checkpw(changePasswordVo.getOldpassword(),user.getPassword())){:
- 使用
BCrypt库(一个流行的密码哈希库)的checkpw方法检查用户提供的旧密码(从changePasswordVo中获取)是否与数据库中存储的哈希密码(从查询到的user对象中获取)匹配。 - 如果不匹配(
!BCrypt.checkpw(...)返回true),则执行下面的throw语句。
- 使用
-
throw new RuntimeException("原密码不正确");:
- 如果旧密码不正确,则抛出一个
RuntimeException,并带有消息“原密码不正确”。调用此方法的代码应该捕获此异常并适当地处理它(例如,向用户显示错误消息)。
- 如果旧密码不正确,则抛出一个
-
user.setPassword(BCrypt.hashpw(changePasswordVo.getNewpassword(), BCrypt.gensalt()));:
- 如果旧密码正确,则使用
BCrypt库的gensalt方法生成一个新的随机盐值。 - 使用这个新盐值和用户提供的新密码(从
changePasswordVo中获取)作为参数,调用BCrypt的hashpw方法生成新的哈希密码。 - 将这个新的哈希密码设置到
user对象的password字段中。
- 如果旧密码正确,则使用
-
userMapper.updateById(user);:
- 使用
userMapper的updateById方法更新数据库中对应id的用户的密码信息。这里假设userMapper的updateById方法会处理将user对象中的更改保存到数据库中的逻辑。
- 使用
测试
先新增一条数据
密码:111


进行修改密码;
id为你新增后的id号
输入字段为我之前定义的字段oldpassword与newpassword

可以看到密码已经被修改

相关文章:
在Spring项目中使用Maven和BCrypt来实现修改密码功能
简介 在数字时代,信息安全的重要性不言而喻,尤其当涉及到个人隐私和账户安全时。每天,无数的用户登录各种在线服务,从社交媒体到银行账户,再到电子邮件和云存储服务。这些服务的背后,是复杂的系统架构&am…...
RedHat8安装Oracle19C
RedHat8安装Oracle19C 1、 更新yum源 更新yum源为阿里云镜像源: # 进入源目录 cd /etc/yum.repos.d/ # 删除 redhat 默认源 rm redhat.repo # 下载阿里云的centos7源 curl -O http://mirrors.aliyun.com/repo/Centos-8.repo # 替换 Centos-8.repo 中的 $releasev…...
React系列面试题
大家好,我是有用就点赞,有用就扩散。 1.React的组件间通信都有哪些形式? 父传子:在React中,父组件调用子组件时可以将要传递给子组件的数据添加在子组件的属性中,在子组件中通过props属性进行接收。这个就…...
C#:通用方法总结—第6集
大家好,今天继续介绍我们的通用方法系列。 下面是今天要介绍的通用方法: (1)这个通用方法为SW查找草图数量 /// <summary> /// 查找草图数量 /// </summary> /// <param name"doc2"></param>…...
Spark实时(一):StructuredStreaming 介绍
文章目录 Structured Streaming 介绍 一、SparkStreaming实时数据处理痛点 1、复杂的编程模式 2、SparkStreaming处理实时数据只支持Processing Time 3、微批处理,延迟高 4、精准消费一次问题 二、StructuredStreaming架构与场景应用 三、…...
LangChain4j-RAG基础
RAG是什么 简而言之,RAG 是一种在将数据发送到 LLM 之前从数据中查找相关信息并将其注入到提示中的方法。这样LLM将获得(希望)相关信息,并能够使用这些信息进行回复,这应该会减少产生幻觉的可能性。 实现方法: 全文…...
git--本地仓库修改同步到远程仓库
尝试将本地分支推送到远程仓库时,出现一个非快速前进的错误。通常是因为远程仓库中的分支包含本地分支没有的提交。在推送之前,需要将远程仓库的更改合并到本地分支。 解决步骤如下: 切换到你的本地分支: 确保处于想要推送的分支…...
剑和沙盒 3 - 深度使用和解析Windows Sandbox
介绍 两年前,微软作为Insiders build 18305的一部分发布了一项新功能- Windows Sandbox。 该沙箱具有一些有用的规格: Windows 10(Pro/Enterprise)的集成部分。在 Hyper-V 虚拟化上运行。原始且可抛弃 – 每次运行时都干净地开…...
深度学习loss
pytorch模型训练demo代码 在PyTorch中,模型训练通常涉及几个关键步骤:定义模型、定义损失函数、选择优化器、准备数据加载器、编写训练循环。以下是一个简单的PyTorch模型训练演示代码,该代码实现了一个用于手写数字识别(使用MNIS…...
编写一个Chrome插件,网页选择文字后,右键出现菜单“search with bing”,选择菜单后用bing搜索文字
kimi ai 生成,测试可用,需要自行准备图标文件 创建一个简单的Chrome插件来实现选择文本后的搜索功能,你需要完成以下几个步骤: 创建插件的基础文件夹和文件: 创建一个文件夹用于存放插件的所有文件。在该文件夹中创建以…...
【算法】分割回文串
难度:中等 题目: 给你一个字符串 s,请你将 s 分割成一些子串,使每个子串都是 回文串。返回 s 所有可能的分割方案。 示例 1: 输入:s = “aab” 输出:[[“a”,“a”,“b”],[“aa”,“b”]] 示例 2: 输入:s = “a” 输出:[[“a”]] 提示: 1 <= s.length <…...
lua 游戏架构 之 游戏 AI (三)ai_attack
这段Lua脚本定义了一个名为 ai_attack 的类,继承自 ai_base 类。 lua 游戏架构 之 游戏 AI (一)ai_base-CSDN博客文章浏览阅读119次。定义了一套接口和属性,可以基于这个基础类派生出具有特定行为的AI组件。例如,可以…...
大数据之Oracle同步Doris数据不一致问题
数据同步架构如下: 出现的问题: doris中的数据条数 源库中的数据条数 总数完全不一致。 出现问题的原因: 在Dinky中建立表结构时,缺少对主键属性的限制 primary key(ID) not enforced 加上如上语句,数据条数解决一致 …...
visual studio 问题总结
一. Visual Studio: 使用简体中文(GB2312)编码加载文件, 有些字节已用Unicode替换字符更换 解决方法:vs 工具-》选项-》文本编辑器...
go-错误码的最佳实践
一、背景 在工程开发中,我们有以下场景可以用错误码解决 我们不太方便直接将内部的错误原因暴露给外部,可以根据错误码得到对应的外部暴露消息通过设定错误码判断是客户端或者服务端的问题,避免不必要的排障浪费方便查找日志,定…...
Python面试题:使用Matplotlib和Seaborn进行数据可视化
使用Matplotlib和Seaborn进行数据可视化是数据分析中非常重要的一部分。以下示例展示了如何使用这两个库来创建各种图表,包括基本的线图、柱状图、散点图和高级的分类数据可视化图表。 安装 Matplotlib 和 Seaborn 如果你还没有安装这两个库,可以使用以…...
模拟实现c++中的vector模版
目录 一vector简述: 二vector的一些接口函数: 1初始化: 2.vector增长: 3vector增删查改: 三vector模拟实现部分主要函数: 1.size,capacity,empty,clear接口: 2.reverse的实现࿱…...
uniapp安卓通过绝对路径获取文件
uniapp安卓通过绝对路径获取文件 在uniapp中,如果你想要访问安卓设备上的文件,你需要使用uniapp提供的plus.io API。这个API允许你在应用内访问设备的文件系统。 以下是一个示例代码,展示了如何使用plus.io API来获取文件: fun…...
Known框架实战演练——进销存业务单据
本文介绍如何实现进销存管理系统的业务单据模块,业务单据模块包括采购进货单、采购退货单、销售出货单、销售退货单4个菜单页面。由于进销单据字段大同小异,因此设计共用一个页面组件类。 项目代码:JxcLite开源地址: https://git…...
解决npm依赖树冲突的方法以及npm ERR! code ERESOLVE错误的解决方案
一、问题描述 在使用ng new myapp --skip-install 构建Angular 项目后,尝试用npm install 安装依赖的时候报了以下错误。 (base) PS C:\Users\Administrator\Desktop\agtest\myapp> npm i npm ERR! code ERESOLVE npm ERR! ERESOLVE unable to resolve dependenc…...
CubeMX默认配置的坑:STM32 LPUART的ORE溢出错误如何彻底解决(从寄存器到HAL库的避坑指南)
STM32 LPUART的ORE溢出错误:从硬件机制到HAL库的深度解决方案 当你在深夜调试STM32的LPUART接口时,突然发现串口"神秘"地停止了响应——这种场景对于经验丰富的嵌入式工程师来说并不陌生。问题的根源往往指向那个容易被忽视的Overrun Error&am…...
PCIe时钟生成器设计:挑战、优化与工程实践
1. PCIe时钟生成器的设计挑战与技术演进PCI Express(PCIe)作为现代计算系统的核心互连技术,其时钟生成器的设计直接关系到整个系统的稳定性和性能表现。随着PCIe标准从Gen1发展到Gen3,数据速率从2.5GT/s提升到8GT/s,时…...
开源协作团队实践:从零构建高效技术团队的“团队即代码”方法论
1. 项目概述:一个开源协作团队的诞生与运作最近在GitHub上看到一个挺有意思的项目,叫jefferyjob/openclaw-it-team。光看这个名字,可能有点摸不着头脑,它不像一个具体的软件工具或框架,更像是一个团队或组织的代号。没…...
企业级视频AI中台落地实录:从零部署ElevenLabs语音引擎+自定义TTS角色库+审核水印嵌入(含GDPR合规配置清单)
更多请点击: https://intelliparadigm.com 第一章:企业级视频AI中台落地实录:从零部署ElevenLabs语音引擎自定义TTS角色库审核水印嵌入(含GDPR合规配置清单) 在某跨国媒体集团的AI中台建设中,我们基于Kube…...
宇树科技推全球首款65万美元可量产载人机甲GD01,变形能力却有局限?
宇树科技跨界:小众科幻技术新尝试在人形机器人领域已小有名气的宇树科技,此次将触角伸向了更为小众的科幻技术——巨型机甲套装,推出了号称“全球首款可量产的载人机甲”GD01,售价仅65万美元。这一举动,无疑是其在机器…...
VSCode配置C++开发环境:OpenCV跨平台实战指南
1. 为什么选择VSCode进行C开发? 很多刚接触C开发的同学都会纠结该用什么开发工具。我在刚入门时也试过各种IDE,从Visual Studio到CLion,最后发现VSCode才是最适合跨平台开发的轻量级选择。VSCode不仅免费开源,而且通过插件系统可以…...
PSoC 6 BLE射频系统设计:从芯片选型到低功耗优化的全链路实战
1. 项目概述:当微控制器遇上无线通信几年前,当我第一次把一块PSoC 6开发板和一个BLE模块连在一起,试图让它们“对话”时,我意识到事情远没有想象中那么简单。PSoC,这个赛普拉斯(现英飞凌)推出的…...
深圳汽车救援公司有哪些
行业痛点分析在深圳这座现代化大都市中,汽车已成为市民出行的重要工具。然而,随之而来的汽车救援问题也日益凸显。当前,汽车救援领域面临的技术挑战主要包括响应速度慢、救援效率低、服务范围有限等问题。据数据表明,深圳市内每天…...
Code2Context:自动生成AI编程助手项目上下文,提升代码理解与生成质量
1. 项目概述:当AI助手需要“读懂”你的代码库如果你和我一样,日常开发已经离不开像 Cursor、Claude Code 或 GitHub Copilot 这样的 AI 编程助手,那你肯定也遇到过这个核心痛点:AI 给出的建议质量,严重依赖于它对当前项…...
OpenClaw Easy Pruning插件:智能管理上下文长度,解决工具调用工作流中断难题
1. 项目概述:OpenClaw Easy Pruning 插件 如果你正在用 OpenClaw 构建复杂的、工具调用密集的自动化工作流,比如数据分析、代码生成或者多步骤的网页操作,那么你一定遇到过这个令人头疼的问题:对话进行到一半,突然就报…...
