笔记项目 day02

news2025/7/18 11:51:02

一、用户登录接口

请求参数:

用loginDTO来封装请求参数,要加上@RequestBody注解

响应参数:

由于data里内容较多,考虑将其封装到一个LoginUser的实体中,用户登陆后,需要生成jwtToken并返回给前端。

登录功能的service层具体实现步骤:

1.根据传进来的account或email到数据库里查,查出该用户的信息。若两者都不存在,则报错。

2.校验用户的是否存在及密码是否正确

3.生成jwt令牌。

4.修改最后登陆时间lastLoginAt

代码如下:

    @Override
    public Result<LoginVO> login(LoginDTO loginDTO) {
        User loginUser = null;
        //1.
        if(loginDTO.getAccount() == null && loginDTO.getEmail() == null){
            return ResultUtil.error("请提供用户名或邮箱!");
        }
        if(loginDTO.getAccount() == null && loginDTO.getEmail() != null){
             loginUser = userMapper.getUserByEmail(loginDTO.getEmail());
        }
        if(loginDTO.getEmail() == null && loginDTO.getAccount() != null){
             loginUser = userMapper.getUserByAccount(loginDTO.getAccount());
        }
        //2.
        if(loginUser == null){
            return ResultUtil.error("用户不存在!请注册!");
        }
        if( !loginDTO.getPassword().equals(loginUser.getPassword())){
            return ResultUtil.error("密码错误!");
        }
        //3.
        String token = jwtUtil.generateToken(loginUser.getUserId());
        loginUser.setLastLoginAt(LocalDateTime.now());
        LoginVO loginVO = new LoginVO();
        BeanUtils.copyProperties(loginUser, loginVO);
        return Result.success(loginVO , token);
    }

二、设计jwt工具类

一个jwt工具类应该包含以下几个功能:

1.生成jwt令牌

流程如下:

其中,claims是载荷,即往jwt令牌里存储的信息,后续可以通过jwt令牌获取这些信息。

使用jwts的builder()链式生成token,需要指定载荷、签发时间、过期时间、签名算法及密钥secret等信息。

2.从前端传来的jwt令牌里读取信息

    public Long getUserIdFromToken(String token) {
        try{Claims claims = Jwts.parser()
                .setSigningKey(secret)
                .parseClaimsJws(token)
                .getBody();
            Long userId = (Long) claims.get("userId");
            return userId;}
        catch(Exception e){
            return null;
        }
    }

一个天坑:

此处jwt再解析时,claims.get("userID")一定要先转换成字符串,再转换成Long,若直接使用(long),由于JWT的声明(claims)基于JSON标准,JSON不区分整数的具体类型(如IntegerLong)。当数值较小时,某些JWT实现(如JJWT)可能将其解析为Integer,而较大的数值解析为Long,直接强制类型转换会导致异常。

因此:使用Long.valueOf()来间接转换。

    public Long getUserIdFromToken(String token) {
        try {
            Claims claims = Jwts.parser()
                    .setSigningKey(secret)
                    .parseClaimsJws(token)
                    .getBody();
            
            return Long.valueOf(claims.get("userId").toString());
        } catch (Exception e) {
            return null;
        }
    }

3.校验jwt令牌是否有效

    //方法3. 检验令牌是否有效
    public boolean validateToken(String token) {
        try {
            Jwts.parser().setSigningKey(secret).parseClaimsJws(token);
            return true;
        } catch (Exception e) {
            return false;
        }
    }

4.刷新jwt令牌

    public String refreshToken(Long userId) {
        return generateToken(userId);
    }

三、获取当前用户信息

请求:无,只需携带jwt令牌

响应如下:跟LoginUserVo是同样的

四、如何从请求中获取jwt令牌?

在第三个需求中,需要先读取jwt令牌,再从jwt令牌中读取userID信息。因此,需要从请求中获取jwt令牌。

这个需求的实现有几种方案:

1.使用threadlocal来存储和传递

2.使用request.setattribute

3.使用@RequestScope

此处选择第三种。

实现步骤:

1.定义 @RequestScope bean,需要加上@Component @RequestScope @Data 这几个注解。

这个类的作用就是每遇到一个个http请求,就会创建它,我们可以将一些数据存进去,以便后续使用。

创建时机​​:每个 HTTP 请求开始时,Spring 会创建一个新的 RequestContext 实例。

​销毁时机​​:请求处理结束后,Spring 自动销毁该实例

@Data
@RequestScope
@Component
public class RequestContext {
    public Long userId;
    public String token;
    public Boolean isLogin;
}

RequestContext中,isLogin的作用:存储当前用户有无登录,若未登录,设置成false,只允许其访问登录和注册的服务。

2.在拦截器中注入并设置数据

3.将拦截器注册到WebConfig中

4. 在 Controller 或 Service 中使用数据​

在需要访问请求级别数据的组件中,直接注入 RequestContext Bean。

五、拦截器怎么写

需求:每当遇到一个请求,就将他拦截下来,并从中读取jwt 令牌。

具体步骤:

1.定义一个类,它必须实现  HandlerInterceptor ,

2.重写preHandle这个方法,里面为 是否放行的逻辑

public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler)
            throws Exception {
        // 对于每个请求进行拦截,获取请求头中的 token
        String token = request.getHeader("Authorization");
        // 然后对 token 进行处理,并将 token 携带的信息存储到,在请求周期中全局存在的 requestScopeData 中

        //0.token为null
        if(token == null){
            requestContext.setToken(null);
            requestContext.setUserId(null);
            requestContext.setIsLogin(false);
            return true; //不管有没有登录,或者jwt令牌合法与否,都放行,后续对这些isLogin为false的请求只开放登陆的权限
        }
        token = token.replace("Bearer", "");
        // 若令牌不合法,requestContext的登录状态设置成false
        if( ! jwtUtil.validateToken(token)){
            requestContext.setIsLogin(false);
        }else{
            requestContext.setIsLogin(true);
            requestContext.setToken(token);
            requestContext.setUserId(jwtUtil.getUserIdFromToken(token));
        }
        return true;
    }

3.配置拦截器

    //配置拦截器
    @Override
    public void addInterceptors(InterceptorRegistry registry) {
        registry.addInterceptor(tokenInterceptor)
                .addPathPatterns("/**")
                .excludePathPatterns("/login", "/error");
    }
}

六、今日踩坑小结

1.请求体若使用实体类封装 , 必须加上@RequestBody , 否则后端接受不到前端传来的数据

2.当前端传来的参数和方法参数不一致时使用@RequestParam注解,当然,前端传来多个参数时必须使用

@GetMapping("/user")
public String getUser(@RequestParam("id") Long userId) {
    // 通过 /user?id=123 访问时,userId=123
    return "user";
}

3.@PathVariable

4.@Param注解:mapper层定义的方法,当接受多个参数时,必须使用

// Mapper接口
public interface UserMapper {
    List<User> findUsersByConditions(
        @Param("name") String name,
        @Param("status") Integer status
    );
}

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

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

相关文章

国鑫主板bios切换显示模式为独立显卡

# 进入到Platform Miscellaneous Configuration Active Video 切换为PCIE Device保存退出&#xff01; 如果之前有安装过nvidia驱动&#xff0c;记得卸载掉再安装一遍。

【日撸 Java 300行】Day 14(栈)

目录 Day 14&#xff1a;栈 一、栈的基本知识 二、栈的方法 1. 顺序表实现栈 2. 入栈 3. 出栈 三、代码及测试 拓展&#xff1a; 小结 Day 14&#xff1a;栈 Task&#xff1a; push 和 pop 均只能在栈顶操作.没有循环, 时间复杂度为 O(1). 一、栈的基本知识 详细的介…

2025最新出版 Microsoft Project由入门到精通(七)

目录 优化资源——在资源使用状况视图中查看资源的负荷情况 在资源图表中查看资源的负荷情况 优化资源——资源出现冲突时的原因及处理办法 资源过度分类的处理解决办法 首先检查任务工时的合理性并调整 增加资源供给 回到资源工作表中双击对应的过度分配资源 替换资…

修改(替换)文件中的指定内容并保留文件修改前的时间(即修改前后文件的最后修改时间保持不变)

&#x1f91f;致敬读者 &#x1f7e9;感谢阅读&#x1f7e6;笑口常开&#x1f7ea;生日快乐⬛早点睡觉 &#x1f4d8;博主相关 &#x1f7e7;博主信息&#x1f7e8;博客首页&#x1f7eb;专栏推荐&#x1f7e5;活动信息 文章目录 修改&#xff08;替换&#xff09;文件中的指…

应用探析|千眼狼PIV测量系统在职业病防治中的应用

1、职业病防治背景 随着《职业病防治法》及各省市“十四五”职业病防治规划的深入推进&#xff0c;工作场所粉尘危害监测与防控已成为疾控部门的核心任务。以矿山、建材、冶金、化工等行业为例&#xff0c;粉尘浓度、分布及传播特性的精准测量是评估职业病风险的关键。 传统的…

nvidia驱动更新-先卸载再安装-ubuntu

显卡驱动升级前&#xff0c;卸载旧版本&#xff0c;可采用两种方式。 1.命令行 &#xff08;1&#xff09;查找已安装的 NVIDIA 驱动和相关包&#xff1a;dpkg -l | grep nvidia &#xff08;2&#xff09;完全卸载 NVIDIA 驱动&#xff1a;sudo apt remove purge nvidia-*…

推荐算法工程化:ZKmall模板商城的B2C 商城的用户分层推荐策略

在 B2C 电商竞争激烈的市场环境中&#xff0c;精准推荐已成为提升用户体验、促进商品销售的关键。ZKmall 模板商城通过推荐算法工程化手段&#xff0c;深度挖掘用户数据价值&#xff0c;制定科学的用户分层推荐策略&#xff0c;实现 “千人千面” 的个性化推荐&#xff0c;帮助…

基于Java和PostGIS的AOI面数据球面面积计算实践

目录 前言 一、计算方法简介 二、球面面积计算 1、AOI数据转Polygon 2、Geotools面积计算 3、GeographicLib面积计算 4、PostGIS面积计算 三、结果分析 1、不同算法结果对比 2、与互联网AOI对比 3、与天地图测面对比 四、总结 前言 在现代地理信息系统&#xff08;G…

Spring Boot之Web服务器的启动流程分析

如何判断创建哪种web容器&#xff1a;servlet&#xff1f;reactive&#xff1f; 我们在启动Spring Boot程序的时候&#xff0c;会使用SpringApplication.run方法来启动&#xff0c;在启动流程中首先要判断的就是需要启动什么类型的服务器&#xff0c;是servlet&#xff1f;或者…

【周输入】510周阅读推荐-1

本号一年了&#xff0c;有一定的成长&#xff0c;也有很多读者和点赞。自觉更新仍然远远不够&#xff0c;需要继续努力。 但是还是要坚持2点&#xff1a; 在当前这个时代&#xff0c;信息大爆炸&#xff0c;层次不齐&#xff0c;不追加多&#xff0c; 信息输入可以很多&#x…

基于动态规划的强化学习方法

目录 # 动态规划 # 基于动态规划的强化学习方法 # 求解过程&#xff1a; ## 策略评估 ## 策略提升 # 价值迭代算法 # 参考 # 动态规划 动态规划的基本思想是将待求解问题分解成若干个子问题&#xff0c;先求解子问题&#xff0c;然后从这些子问题的解得到目标问题的解。…

Uniapp Android/IOS 获取手机通讯录

介绍 最近忙着开发支付宝小程序和app&#xff0c;下面给大家介绍一下 app 获取通讯录的全部过程吧&#xff0c;也是这也是我app开发中的一项需求吧。 效果图如下 勾选配置文件 使用uniapp开发的童鞋都知道有一个配置文件 manifest.json 简单的说一下&#xff0c;就是安卓/ios/…

2025年RIS SCI2区,改进白鲸优化算法+复杂非线性方程组求解,深度解析+性能实测

目录 1.摘要2.白鲸优化算法BWO原理3.改进策略4.结果展示5.参考文献6.代码获取7.读者交流 1.摘要 本文提出了一种改进白鲸优化算法&#xff08;ABWOA&#xff09;用来解决非线性方程组&#xff08;SNLEs&#xff09;求解问题。ABWOA引入了平衡因子和非线性自适应参数&#xff0…

Java后端开发day48--反射动态代理

&#xff08;以下内容全部来自上述课程&#xff09; 反射 反射允许对成员变量&#xff0c;成员方法和构造方法的信息进行编程访问。 就是获取里面的成员变量、构造方法和成员方法&#xff0c;idea中打代码跳出来的提示就是反射。 1. 获取class对象的三种方式 Class.for…

ValueError: Caught ValueError in DataLoader worker process 0.

参考链接&#xff1a; https://stackoverflow.com/questions/1841565/valueerror-invalid-literal-for-int-with-base-10 它提示我有个地方值错误空字符 果然因为格式处理没有传进去东西&#xff0c;找下原因&#xff0c;让它正常处理 原来是相对路径的.影响了程序运行 将v…

【数据结构】——链表OJ(下)

前面我们已经刷了几道单链表的题目&#xff0c;下面我们继续看几道题目。 一、相交链表 这道题题目的要求是很好理解的&#xff0c;就是现在我们有两个链表&#xff0c;然后我们就相办法进行判断&#xff0c;这两个链表是否是相交的&#xff0c;那么链表的相交其实就是有没有共…

Adobe Acrobat pro在一份PDF中插入空白页

在Adobe Acrobat pro中先打开我们的PDF文件&#xff1b; 用鼠标点击需要插入空白页处的上一页&#xff1b; 然后如下图操作&#xff1a; 默认会在光标处的下一页插入一张空白页&#xff0c;你也可以修改插入页的页码或者向前一页插入

java-----异常

对于Error&#xff1a;表示系统级错误或者资源耗尽的状况&#xff0c;像OutOfMemoryError、StackOverflowError等。这类错误是程序无法处理的&#xff0c;通常也不应该尝试去处理。 对于Exception&#xff1a;表示程序可以处理的异常。它又能细分为&#xff1a; 受检查异常&a…

[工具]B站缓存工具箱 (By 郭逍遥)

&#x1f4cc; 项目简介 B站缓存工具箱是一个多功能的B站缓存工具&#xff0c;包含视频下载、缓存重载、文件合并及系统设置四大核心功能。基于yutto开发&#xff0c;采用图形化界面操作&#xff0c;极大简化B站资源获取与管理流程。 工具可以直接将原本缓存的视频读取&#…

自动化测试与功能测试详解

&#x1f345; 点击文末小卡片&#xff0c;免费获取软件测试全套资料&#xff0c;资料在手&#xff0c;涨薪更快 什么是自动化测试? 自动化测试是指利用软件测试工具自动实现全部或部分测试&#xff0c;它是软件测试的一个重要组成 部分&#xff0c;能完成许多手工测试无…