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

 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令牌的生成(有代码)

引入依赖

<dependency>
    <
groupId>io.jsonwebtoken</groupId>
    <
artifactId>jjwt</artifactId>
    <
version>0.9.1</version>
</
dependency>

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
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

dream_ready

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值