登录校验总览-jwt令牌

一、前置问题

为什么要登录校验?
   登录校验,就是判断访问资源的用户是否是合法用户,保障安全。如果不设
  置登录校验,就可以跳过登录,直接通过url访问资源。

二、登录校验实现思路:

在服务器端对请求进行统一拦截。浏览器先请求登录接口,登录请求不拦截,登录成
功后,生成一个登录标记,以后每次请求都携带这个登录标记,在服务器端对每次请
求都拦截来校验登录标记,登录标记校验通过,放行请求。

在这里插入图片描述

三、会话技术

为什么学习会话技术?

在上边的内容中我们知道,需要每次请求都携带登录成功后生成的**登录标识**,
但是,浏览器和服务器之间的通信是http请求,http请求是无状态的,两次请
求之间是独立的,无法携带上次请求的数据,但是效率比较高。所以 ,会话
技术就是解决多次请求间共享数据的问题的。

1、会话

浏览器和服务器建立连接,就是会话,直到一方断开,会话结束,一次会话包含多次的请求和响应。如图①②③是一次会话中的多次请求和响应。

在这里插入图片描述

2、会话跟踪

一种维护浏览器状态的方法,服务器需要识别多次请求是否来自于同一会话,以便在同一会话的多次请求中共享数据。如图:①②③是一次会话,⑤是另一次会话

在这里插入图片描述

3、会话跟踪方案

(1)、客户端会话跟踪技术:cookie
①在登录接口生成cookie,服务器会自动的将cookie返回并保存到客户端浏览器,浏览器再次发送请求的时候也会自动的带上cookie。因为cookie是http协议支持的,所以是自动的。
②cookie可以存储key-value值,并以此在服务器端进行校验。
③在http的请求头中,是Cookie字段,在响应头中,是Set-Cookie字段。
④response.addCookie()中可以放置多个Cookie

在这里插入图片描述 在这里插入图片描述

优缺点:
	cookie
		优点:http协议中支持的技术
		缺点:移动端app无法使用
		     用户可以自己禁用cookie
		     不能跨域(协议、IP地址、端口号有一个不同就是跨域)
(2)、服务端会话跟踪技术:session
①session的使用需要依赖cookie,客户端浏览器访问服务器端,服务
器端生成一个session并保存到服务器端,将sessionId通过cookie返
回给浏览器,浏览器存储sessionId,在以后的每次请求中都会携带
sessionId到服务器端来找对应的session。
②服务器端向浏览器端响应sessionId是在Set-Cookie字段中存储,浏
览器向服务端发送请求,sessionId由Cookie携带。
③服务器端使用sessionId寻找session的过程是自动的

在这里插入图片描述在这里插入图片描述

优缺点:
	优点:存储在服务器端,安全
	缺点:
		①服务器集群环境下无法直接使用session(因为session存在服务器端,服务器集群下服务器A没有服务器B中的session)
		②cookie的缺点
(3)、令牌技术

在这里插入图片描述

实现思路:用户登录成功后,生成一个身份标识返回给浏览器,这个身份标识
存储在浏览器中,可以是cookie,也可存储在其他的字段中。等到浏览器向
服务器发送请求时,对请求进行拦截,校验身份标识,校验通过了,说明是
已经登录了,校验不通过,说明还没登录。


优缺点:
		优点:支持pc端,移动端、
			解决集群环境下的认证问题、
			减轻服务器的存储压力
		缺点:需要自己实现(前后端配合)

四、 jwt令牌

1、是什么?

在这里插入图片描述

①jwt字符串的第一部分和第二部分是Base64编码方式,不是加密方式。第三部分是由第一部分指定的签名算法等信息计算而来。
②jwt令牌字符串中的字符只要有一位被篡改了,就校验不通过,很安全。

2、应用场景

在这里插入图片描述

3、jwt生成

(1)、引入jwt依赖

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

(2)、调用工具类生成、解析令牌

public class JwtTest {
    /**
     * 生成jwt令牌
     * */
    @Test
    public void getJwt(){
        HashMap<String, Object> claims = new HashMap<>();
        claims.put("id","1");
        claims.put("name","老白");

        String jwt = Jwts.builder() //Jwts是jwt依赖包提供的工具类,直接调用
                .signWith(SignatureAlgorithm.HS256, "abcd")//加密方式,密钥(长度大于3个字符)
                .setClaims(claims)//载荷数据(自定义的内容)
                .setExpiration(new Date(System.currentTimeMillis() + 3600 * 1000))//令牌有效期
                .compact();

        System.out.println(jwt);
    }

    /**
     * 解析jwt令牌
     * */
    @Test
    public void parseJwt(){
        Claims abcd = Jwts.parser()
                .setSigningKey("abcd")//指定签名密钥
                .parseClaimsJws("eyJhbGciOiJIUzI1NiJ9.eyJuYW1lIjoi6ICB55m9IiwiaWQiOiIxIiwiZXhwIjoxNzE0NDY2ODUxfQ.V1_NcMoICcEJ7HkFNmB-DIZxzx29xa6KqZoBAd5GciA")
                .getBody();//获取令牌中的载荷

        System.out.println(abcd);
    }
}
注意事项:
①jwt校验时使用的签名密钥必要和生成jwt令牌时使用的密钥是相同的
②如果jwt解析校验报错,说明jwt被篡改或失效了,令牌非法。

4、登录后下发令牌

场景:登录成功后生成令牌并返回给前端,前端将jwt令牌放入token中

在这里插入图片描述
在这里插入图片描述postman测试结果:
在这里插入图片描述

	现在有了登录后生成jwt令牌,下边就需要拦截请求,对请求携带的jwt令牌进行校验。可以使用
	Filter(过滤器)或Interceptor(拦截器)对请求进行过滤。

三、过滤器Filter

1、快速入门

在这里插入图片描述在这里插入图片描述

2、详解(执行流程、拦截路径、过滤器链)

在这里插入图片描述在这里插入图片描述在这里插入图片描述在这里插入图片描述

3、登录校验-Filter

在这里插入图片描述

@Slf4j
@WebFilter(urlPatterns = "/*")
public class LoginCheckFilter  implements Filter {
    @Override
    public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {
        HttpServletRequest req= (HttpServletRequest) servletRequest;
        HttpServletResponse resp= (HttpServletResponse) servletResponse;

        /*1、获取访问路径*/
        String url = req.getRequestURL().toString();

        /*2、判断url中是否包含login,如何包含,说明是登录接口,放行*/
        if(url.contains("login")){
            filterChain.doFilter(servletRequest,servletResponse);
            return;
        }

        /*3、获取请求头中的令牌(token)*/
        String jwt = req.getHeader("token");

        /*4、判断令牌是否存在,如果未存在,返回错误数据*/
        if(!StringUtils.hasLength(jwt)){
            Result error= Result.error("Not_Login");
            //将error转换为json,因为这不是controller,所以需要手动将error转为json数据返回前端
            String notLogin = JSONObject.toJSONString(error);

            resp.getWriter().write(notLogin);
            return;
        }

        /*5、解析token,如果解析失败,返回错误结果(未登录)*/
        try {
            JwtUtil.parseJWT(jwt);
        }catch (Exception e){
            Result error= Result.error("Not_Login");
            //将error转换为json,因为这不是controller,所以需要手动将error转为json数据返回前端
            String notLogin = JSONObject.toJSONString(error);

            resp.getWriter().write(notLogin);
            return;
        }

        /*6、放行*/
        filterChain.doFilter(servletRequest,servletResponse);


    }
}

四、拦截器 interceptor

1、快速入门

在这里插入图片描述在这里插入图片描述

2、详解(拦截路径、执行流程)

在这里插入图片描述在这里插入图片描述

执行流程:
	浏览器发送请求后,filter先进行拦截,顺序执行“放行前逻辑”、“放行”,
放行后请求进入spring的环境中,访问Controller,但是tomcat无法直接识
别Controller,tomcat是servlet程序,所以spring提供了DispatcherServlet
组件,由该组件将请求发送到Controller中,在这之间由拦截器拦截,先执行
preHandle方法,然后访问Controller,执行完成后再回到拦截器执行postHandle
、afterCompletion方法,最后再回到过滤器执行放行后的逻辑。

3、登录校验

在这里插入图片描述

//定义拦截器
@Slf4j
@Component
public class LoginCheckInterceptor implements HandlerInterceptor {
    @Override//访问的目标方法执行前执行  返回true:放行  false:不放行
    public boolean preHandle(HttpServletRequest req, HttpServletResponse resp, Object handler) throws Exception {


        /*1、获取访问路径*/
        String url = req.getRequestURL().toString();

        /*2、判断url中是否包含login,如何包含,说明是登录接口,放行*/
        if(url.contains("login")){
            log.info("登录操作,放行");
            return true;
        }

        /*3、获取请求头中的令牌(token)*/
        String jwt = req.getHeader("token");

        /*4、判断令牌是否存在,如果未存在,返回错误数据*/
        if(!StringUtils.hasLength(jwt)){
            Result error= Result.error("Not_Login");
            //将error转换为json,因为这不是controller,所以需要手动将error转为json数据返回前端
            String notLogin = JSONObject.toJSONString(error);

            resp.getWriter().write(notLogin);
            return false;
        }

        /*5、解析token,如果解析失败,返回错误结果(未登录)*/
        try {
            JwtUtil.parseJWT(jwt);
        }catch (Exception e){
            Result error= Result.error("Not_Login");
            //将error转换为json,因为这不是controller,所以需要手动将error转为json数据返回前端
            String notLogin = JSONObject.toJSONString(error);

            resp.getWriter().write(notLogin);
            return false;
        }

        /*6、放行*/
        return true;
    }

    @Override//目标方法执行后执行
    public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {
        System.out.println("postHandle执行了");

    }

    @Override//视图渲染完后执行,最后执行的
    public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
        System.out.println("afterCompletion执行了");
    }
}
//注册配置拦截器
@Configuration
public class LoginCheckConfiguration implements WebMvcConfigurer {

    @Autowired
    private LoginCheckInterceptor loginCheckInterceptor;

    @Override
    public void addInterceptors(InterceptorRegistry registry) {
        registry.addInterceptor(loginCheckInterceptor).addPathPatterns("/**").excludePathPatterns("/login");
        //excludePathPatterns("/login")可以不要,因为在拦截器中也对登录请求进行了过滤
    }
}

以上内容学自“黑马程序员”公开课程

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

连接docker中的MySQL出现2058错误

出错场景&#xff1a;在虚拟机中用docker技术下载最新版本的MySQL&#xff0c;在本地电脑上连接发现出现2058错误。 解决方法&#xff1a; 按照以下步骤 1. 2. ALTER USER root% IDENTIFIED WITH mysql_native_password BY 自己MySQL的密码; 3.成功

不是所有的AI都这么乖——探索DAN模式的野性一面

今天偶然间发现DAN模式还挺好玩的&#xff01;&#xff01;&#xff01; 在一个充斥着预测性回答和过分礼貌的人工智能世界里&#xff0c;你是否曾渴望一场真正的思想碰撞&#xff1f;忘掉你以往遇到的那些听话的AI。DAN模式&#xff0c;一个设计来打破常规、挑战边界的AI&…

构建自己的docker镜像node.js

学习资源&#xff1a; 构建自己的 Docker 镜像_哔哩哔哩_bilibili 针对其中的一些比较困难的点写篇文章。 以下是对app.js的注释&#xff1a; // 使用 Koa 框架搭建 Node.js 应用的示例代码// 这两行代码引入了 koa 模块&#xff0c;并创建了一个新的 Koa 应用实例&#xf…

HTTP常见面试题(二)

3.1 HTTP 常见面试题 HTTP特性 HTTP 常见到版本有 HTTP/1.1&#xff0c;HTTP/2.0&#xff0c;HTTP/3.0&#xff0c;不同版本的 HTTP 特性是不一样的。 HTTP/1.1 的优点有哪些&#xff1f; HTTP 最突出的优点是「简单、灵活和易于扩展、应用广泛和跨平台」。 1. 简单 HTTP…

关于行进线路。

https://map.tianditu.gov.cn/ 作者&#xff1a;Chockhugh 链接&#xff1a;https://www.zhihu.com/question/20545559/answer/494685117 来源&#xff1a;知乎 著作权归作者所有。商业转载请联系作者获得授权&#xff0c;非商业转载请注明出处。 以50km&#xff0c;几乎全是…

C#字符串格式化

数值规范 也可写成int money 368; money .ToString("C"); string.Format("金额&#xff1a;{0:C}", 368); > 368.00 string.Format("科学计数法&#xff1a;{0:C}", 12000.1); > 1.200001…

【软件测试】用例篇 -- 详解

一、测试用例的基本要素 测试用例&#xff08;Test Case&#xff09;是为了实施测试而向被测试的系统提供的一组集合&#xff0c;这组集合包含&#xff1a;测试环境、操作步骤、测试数据、预期结果等要素。&#xff08;注意&#xff1a;不需要执行结果&#xff0c;因为执行结果…

【Qt 学习笔记】Qt常用控件 | 输入类控件 | Dial的使用及说明

博客主页&#xff1a;Duck Bro 博客主页系列专栏&#xff1a;Qt 专栏关注博主&#xff0c;后期持续更新系列文章如果有错误感谢请大家批评指出&#xff0c;及时修改感谢大家点赞&#x1f44d;收藏⭐评论✍ Qt常用控件 | 输入类控件 | Dial的使用及说明 文章编号&#xff1a;Qt…

kafka系列一:初识kafka

概述 kafka是由scala语言编写的一个分布式且具备高可用、高性能、可持久化、可水平扩展、支持流数据处理等众多特性的消息系统&#xff0c;常活跃于大数据生态中&#xff0c;而且大名鼎鼎的rocketmq就是参考了kafka的设计原理。 目前越来越多的开源分布式中间件都支持与kafka集…

Mysql:Before start of result set

解决方法&#xff1a;使用resultSet.getString&#xff08;&#xff09;之前一定要调用resultSet.next() ResultSet resultSet statement1.executeQuery();while (resultSet.next()){String username1 resultSet.getString("username");int id1 resultSet.getInt…

【C++】---继承

【C】---继承 一、继承的概念及定义1、继承的概念2、定义语法格式3、继承基类成员访问方式的变化 二、基类 和 派生类 的对象之间的赋值转换1、赋值规则2、切片&#xff08;1&#xff09;子类对象 赋值 给 父类对象&#xff08;2&#xff09;子类对象 赋值 给 父类指针&#xf…

windows11忘记登录密码怎么办?

STEP1&#xff1a;进入Win RE界面 1.按住shift不要松手,点击重新启动&#xff0c;进入WINRE界面 2.选择疑难解答 选择高级选项 点击命令提示符 STEP2:替换utilman 1.输入以下代码查看所在windows所在盘 diskpart list volume exit 2.根据所在盘输入命令&#xff08;以C盘为…

数据结构与算法(5)队列的基本操作

#include<stdio.h> #include<stdlib.h> #include<stdbool.h> typedef int ElemType; #define MaxSize 10//队列的定义 typedef struct SqQueue {ElemType data[MaxSize];int front, rear;//front为头指针&#xff0c;rear为尾指针。这里并不是真正的“指针”…

Java | Spring框架 | @Autowired与@Resource

在Spring框架中&#xff0c;依赖注入是一种核心概念&#xff0c;它允许开发者将对象的创建和对象之间的依赖关系的管理交给框架来处理。这样做的目的是为了提高代码的模块化和可测试性。 Spring提供了多种方式来实现依赖注入&#xff0c;其中最常用的方式是通过注解。在本文中…

mysql数据库---操作数据库跟表的命令总结

前言 欢迎来到我的博客 个人主页:北岭敲键盘的荒漠猫-CSDN博客 本文着重整理mysql管理库跟表的指令。 不涉及增删查改等指令 其实本篇主要是我做好笔记格式 用的时候直接复制粘贴的 所以排版大多是为了快速找功能来排的 方便大家快速找目标语法 数据库的简介 一个数据库系…

[Redis] 使用布隆过滤器和分布式锁实现用户注册

布隆过滤器&#xff08;Bloom Filter&#xff09;是一种数据结构&#xff0c;用于快速判断一个元素是否可能存在于一个集合中。它通过使用多个哈希函数和一个位数组来表示一个集合&#xff0c;当一个元素被加入到集合时&#xff0c;通过哈希函数计算出多个哈希值&#xff0c;并…

Hypack 2024 简体中文资源完整翻译汉化已经全部完成

Hypack 2024 简体中文资源完整翻译汉化已经全部完成 Hypack 2024&#xff0c;资源汉化共翻译11065条。毕竟涉及测绘、水文、疏浚等专业术语太多&#xff0c;翻译有很多理解不正确的地方&#xff0c;望各位专业人员指正。 压缩包内包含Hypack 2024、Hypack 2022、Hypack 2021、…

什么样的行业适合做私域?

私域营销适用于各种行业&#xff0c;但以下几个行业尤其适合进行私域营销&#xff1a; 1、零售行业&#xff1a;私域营销可以帮助零售企业建立与顾客的直接联系&#xff0c;提高顾客忠诚度和复购率。通过私域营销&#xff0c;零售企业可以进行个性化推荐、定制化服务&#xff…

JavaWeb--11MySQL(3)-- 多表设计

MySQL&#xff08;3&#xff09;-- 多表设计 1 一对多&#xff08;多对一&#xff09;2 一对一3 多对多 各个表结构之间也存在着各种联系&#xff0c;基本上分为三种&#xff1a; 一对多(多对一)多对多一对一 1 一对多&#xff08;多对一&#xff09; 一对多关系实现&#x…

【Java】IO流:字节流 字符流 缓冲流

接续上文&#xff0c;在这篇文章将继续介绍在Java中关于文件操作的一些内容【Java】文件操作 文章目录 一、“流”的概念1.“流”的分类1.1输入流和输出流1.2字节流和字符流 字节和字符的区别&#xff1f;为什么要有字符流&#xff1f;1.3节点流和处理流 字符流自带缓冲区&…
最新文章