JWT令牌基础教程 全方位带你剖析JWT令牌,在Springboot中使用JWT技术体系,完成拦截器的实现 Interceptor (后附源码)

简介: 文章介绍了JWT令牌的基础教程,包括其应用场景、组成部分、生成和校验方法,并在Springboot中使用JWT技术体系完成拦截器的实现。

JWT官网:http://https:/jwt.io/

1、JWT应用场景:登陆验证

第一次登录生成令牌,之后每次前端向后端发送请求都要带上JWT令牌,用来校验

若令牌出错或没有,则会被拦截,无法到达后端接口进行处理

2、JWT组成

实际生成如下:

JWT由一段字符组成,这些字符以两个英文点符号作为分割

1、Header(头部)中:

    alg指定数字签名算法,此处指定数字签名算法为HS256算法

    type指定令牌类型,此处指定令牌类型为JWT

2、Payload(载荷)就是存储自定义的信息:

    此处的name以及iat都是我们自定义的信息

头部和载荷原始数据都是JSON格式的数据,在JWT令牌生成中基于Base64格式进行编码

Signature(数字签名)部分将头部和载荷结合起来,并加入指定密钥(自己指定,大多为自己指定的字符串),采用alg中声明的数字签名算法进行加密,不基于Base64格式进行编码

3、JWT令牌的生成(有代码)

引入依赖


io.jsonwebtoken
jjwt
0.9.1

JWT令牌的生成代码

    public void testGenjwt(){
        Map<String, Object> claims = new HashMap<>();
        claims.put("id", 1);
        claims.put("name", "tom");
        String jwt = Jwts.builder()
                .signWith(SignatureAlgorithm.HS256, "itdream_ready")  // 签名算法
                .setClaims(claims)     // 自定义内容(载荷)
                .setExpiration(new Date(System.currentTimeMillis() + 3600 * 1000))   // 设置有效期为1h  +号右侧数字单位是毫秒
                .compact();   // 对生成的内容进行字符串的拼接
        System.out.println(jwt);
    }

JWT令牌的校验

此处的.parseClaimJws参数对应生成的JWT令牌

    public void testParseJwt(){
        Claims claims = Jwts.parser()
                .setSigningKey("itdream_ready")      // 指定密钥
                // 下面括号内填入生成的JWT令牌
                .parseClaimsJws("eyJhbGciOiJIUzI1NiJ9.eyJuYW1lIjoidG9tIiwiaWQiOjEsImV4cCI6MTY5NzU0ODc5Mn0.6iT6BrK3lKZbYrCmm_5_Y6py_vLF25_YMm3MtodsmPA")
                .getBody();// 拿到 body 中的部分
        System.out.println(claims);
    }

登录 —— 生成JWT令牌

此处代码无法直接运行,仅供参考

4、拦截器(Interceptor)

拦截器概述

定义拦截器

此处只是定义,若想定义的拦截器发挥功能,需注册拦截器

定义一个类,实现HandlerInterceptor接口,并重写对应方法

一般来说preHandle是必须要重写的,因为它在请求到达目标资源方法前期执行,是拦截器的核心

在preHandle中进行JWT令牌的校验,并根据校验结果选择放行或拦截

参考代码(无法执行,只供参考):

    /**
     * 校验jwt
     *
     * @param request
     * @param response
     * @param handler
     * @return
     * @throws Exception
     */
    // 该方法在请求处理方法之前被自动调用
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {

        System.out.println("当前线程的id:" + Thread.currentThread().getId());

        //判断当前拦截到的是Controller的方法还是其他资源
        if (!(handler instanceof HandlerMethod)) {
            //当前拦截到的不是动态方法,直接放行
            return true;
        }

        //1、从请求头中获取令牌
        String token = request.getHeader(jwtProperties.getAdminTokenName());

        //2、校验令牌
        try {
            log.info("jwt校验:{}", token);
            Claims claims = JwtUtil.parseJWT(jwtProperties.getAdminSecretKey(), token);
            Long empId = Long.valueOf(claims.get(JwtClaimsConstant.EMP_ID).toString());
            log.info("当前员工id:", empId);
            // 将发送该请求的管理员id记录下来,保存下来
            BaseContext.setCurrentId(empId);
            //3、通过,放行
            return true;
        } catch (Exception ex) {
            //4、不通过,响应401状态码
            response.setStatus(401);
            return false;
        }
    }

注册拦截器 —— 拦截路径

将定义的拦截器给予启用,并设置拦截和放行的路径

参考代码(无法直接运行,只供参考):

    /**
     * 注册自定义拦截器
     *
     * @param registry
     */
    protected void addInterceptors(InterceptorRegistry registry) {
        // log.info("") 是一个日志输出语句,通常在代码中用于输出日志信息
        log.info("开始注册自定义拦截器...");
        // 设置什么路径的请求该拦什么不该拦
        registry.addInterceptor(jwtTokenAdminInterceptor)
                .addPathPatterns("/admin/**")  // 指定拦截的资源路径
                .excludePathPatterns("/admin/employee/login");   // 指定不需要拦截的资源路径
        registry.addInterceptor(jwtTokenUserInterceptor)
                .addPathPatterns("/user/**")
                .excludePathPatterns("/user/user/login")
                .excludePathPatterns("/user/shop/status");
    }
目录
相关文章
|
2月前
|
JSON 安全 算法
|
1月前
|
数据采集 监控 前端开发
二级公立医院绩效考核系统源码,B/S架构,前后端分别基于Spring Boot和Avue框架
医院绩效管理系统通过与HIS系统的无缝对接,实现数据网络化采集、评价结果透明化管理及奖金分配自动化生成。系统涵盖科室和个人绩效考核、医疗质量考核、数据采集、绩效工资核算、收支核算、工作量统计、单项奖惩等功能,提升绩效评估的全面性、准确性和公正性。技术栈采用B/S架构,前后端分别基于Spring Boot和Avue框架。
|
2月前
|
Cloud Native Java C++
Springboot3新特性:开发第一个 GraalVM 本机应用程序(完整教程)
文章介绍如何在Spring Boot 3中利用GraalVM将Java应用程序编译成独立的本机二进制文件,从而提高启动速度、减少内存占用,并实现不依赖JVM运行。
233 1
Springboot3新特性:开发第一个 GraalVM 本机应用程序(完整教程)
|
2月前
|
人工智能 自然语言处理 前端开发
SpringBoot + 通义千问 + 自定义React组件:支持EventStream数据解析的技术实践
【10月更文挑战第7天】在现代Web开发中,集成多种技术栈以实现复杂的功能需求已成为常态。本文将详细介绍如何使用SpringBoot作为后端框架,结合阿里巴巴的通义千问(一个强大的自然语言处理服务),并通过自定义React组件来支持服务器发送事件(SSE, Server-Sent Events)的EventStream数据解析。这一组合不仅能够实现高效的实时通信,还能利用AI技术提升用户体验。
184 2
|
1月前
|
JavaScript Java 项目管理
Java毕设学习 基于SpringBoot + Vue 的医院管理系统 持续给大家寻找Java毕设学习项目(附源码)
基于SpringBoot + Vue的医院管理系统,涵盖医院、患者、挂号、药物、检查、病床、排班管理和数据分析等功能。开发工具为IDEA和HBuilder X,环境需配置jdk8、Node.js14、MySQL8。文末提供源码下载链接。
|
2月前
|
存储 安全 Java
|
2月前
|
Java API Apache
Springboot+shiro,完整教程,带你学会shiro
这篇文章提供了一个完整的Apache Shiro与Spring Boot结合使用的教程,包括Shiro的配置、使用以及在非Web和Web环境中进行身份验证和授权的示例。
69 2
Springboot+shiro,完整教程,带你学会shiro
|
2月前
|
前端开发 Java Apache
Springboot整合shiro,带你学会shiro,入门级别教程,由浅入深,完整代码案例,各位项目想加这个模块的人也可以看这个,又或者不会mybatis-plus的也可以看这个
本文详细讲解了如何整合Apache Shiro与Spring Boot项目,包括数据库准备、项目配置、实体类、Mapper、Service、Controller的创建和配置,以及Shiro的配置和使用。
371 1
Springboot整合shiro,带你学会shiro,入门级别教程,由浅入深,完整代码案例,各位项目想加这个模块的人也可以看这个,又或者不会mybatis-plus的也可以看这个
|
2月前
|
缓存 NoSQL Java
springboot的缓存和redis缓存,入门级别教程
本文介绍了Spring Boot中的缓存机制,包括使用默认的JVM缓存和集成Redis缓存,以及如何配置和使用缓存来提高应用程序性能。
115 1
springboot的缓存和redis缓存,入门级别教程
|
2月前
|
JSON NoSQL Java
springBoot:jwt&redis&文件操作&常见请求错误代码&参数注解 (九)
该文档涵盖JWT(JSON Web Token)的组成、依赖、工具类创建及拦截器配置,并介绍了Redis的依赖配置与文件操作相关功能,包括文件上传、下载、删除及批量删除的方法。同时,文档还列举了常见的HTTP请求错误代码及其含义,并详细解释了@RequestParam与@PathVariable等参数注解的区别与用法。