token系统讲解及过期处理

简介: 笔记

40.png


这玩意很简单,记录一下吧,给入门的小白用下


1. token是什么?用来做什么



token通常译为令牌,暗号。举个例子:我是古时候皇城中大内的高手(看门的大爷),进皇宫需要皇帝发的特制令牌,没有指定的令牌是不能进的,我会把你拦住,说不定还会把你胖揍一顿。皇宫在这里就相当于我们项目,为了项目的安全性,设置了这个token,进项目的人都需要有这个玩意,证明你有这个权限。

token是怎么生成的? 通过哈希算法(不常用)、uuid(雪花算法,可保持全球唯一性),jwt工具等生成随机字符串(也可能会拼接上用户的一些信息)

为什么token要有有效期?而且设置得这么短? 为了提高token的安全性,不能永久有效,降低被别人劫持的风险

有时候登录的时候为什么会有多个token?

1. 常规 token:使用频繁,有效期比较短

2. refresh_token:当token过期的时候,用来得到新的token的,使用不频繁,有效期比较长(一般为一个月)

我们怎么知道token过期了? 通过调接口时返回的状态码,如果状态码是401(一般情况下),就说明没有携带token,或者token过期了


41.png


2. token存储在哪?过期了怎么办?


token由服务端生成,通过接口传给前端。

一般在登录接口会给token值。

存储看具体的项目场景,一般存储在本地缓存里,有两种方案供君选择:

sessionStorage(会话级别,网页一关就歇逼了)

localStorage(永久的,需要手动去清除)

token过期处理

重新登录 => 清空token,跳转登录页

refresh_token => 得到一个全新的token


3. 请求拦截与响应拦截执行时机(面试重点)


这玩意面试问的多,看图很清晰:

42.png请求拦截器执行时机:axios调用之后,浏览器真正发起请求之前

响应拦截器执行时机:浏览器接受到响应数据之后,axios拿到数据之前


4. 解决token过期方案一: 重新登录


知道出现了401的错误: 使用响应拦截器

进行页面的跳转

此种方案用户体验不是很好(目前大部分企业及项目都这么干)

import store from '@/store'
// 响应拦截器
request.interceptors.response.use(
  function (response) {
    // 响应成功(响应状态码是 2xx)时执行第一个回调函数
    console.log('响应成功....')
    return response
  }, function (error) {
    // 网络异常或响应失败(响应状态码是非2开头)时执行第二个回调函数
    console.log('响应失败....')
    // 1. 判断响应的状态码是不是401,如果不是401就不用做任何处理
    if (error.response && error.response.status === 401) {
      // 2. 如果是401,就先清空token, 并跳转到登录页
      store.commit('setUser', null)
      router.push('/login')
    }
    return Promise.reject(error)
  }
)


5. 方案二:使用 refresh_token 方案


判断有没有refresh_token,如果没有跳转登录页

如果有refresh_token,调用刷新token的接口,拿到新的token

更新vuex和loaclStoreage存储的token

为原来调用接口方,重新去调用接口

如果利用refresh_token,刷新token的接口失败,则还是跳转到登录页

附上了详细的注释,讲道理大佬们应该看得懂

import axios from 'axios'
import store from '@/store'
import router from '@/router'
const baseURL = 'http://xxx.xxx.net/'
const request = axios.create({
  baseURL // 接口的基准路径
})
// 请求拦截器
// Add a request interceptor
request.interceptors.request.use(function (config) {
  // 请求发起会经过这里
  // config:本次请求的请求配置对象
  const { user } = store.state
  if (user && user.token) {
    config.headers.Authorization = `Bearer ${user.token}`
  } 
  // 注意:这里务必要返回 config 配置对象,否则请求就停在这里出不去了
  return config
}, function (error) {
  // 如果请求出错了(还没有发出去)会进入这里
  return Promise.reject(error)
})
request.interceptors.response.use(
  function (response) {
    // 响应成功(响应状态码是 2xx)时执行第一个回调函数
    return response
  }, async function (error) {
    // 网络异常或响应失败(响应状态码是非2开头)时执行第二个回调函数
    console.log('响应失败时执行的代码....')
    if (error.response && error.response.status === 401) {
      const user = store.state.user
      if (user && user.refresh_token) {
        // 1. 判断有没有refresh_token,如果没有跳转登录页
        // 2. 如果有refresh_token,调用刷新token的接口,拿到新的token
        try {
          var res = await axios({
            baseURL: baseURL,
            method: 'PUT',
            url: '/xxx/xxx',
            headers: {
              Authorization: `Bearer ${user.refresh_token}`
            }
          })
        } catch {
          // 5. 如果refresh_token过期,进行清空token并跳转到登录页的处理
          store.commit('setUser', null)
          router.push('/login')
        }
        //  3. 更新vuex和loaclStoreage存储的token
        store.commit('setUser', {
          refresh_token: user.refresh_token,
          token: res.data.data.token
        })
        //    4. 为原来调用接口方,重新去调用接口
        console.log(error.config)
        return request(error.config)
      } else {
        // 1. 判断有没有refresh_token,如果没有跳转登录页
        store.commit('setUser', null)
        router.push('/login')
      }
    }
    // 3. 如果利用refresh_token,刷新token的接口失败,则还是跳转到登录页
    // eslint-disable-next-line prefer-promise-reject-errors
    return Promise.reject(error)
  }
)
export default request


目录
相关文章
|
物联网
低功耗蓝牙(BLE)设备常用的4种角色
对于主从设备的其它说法,大家需要了解一下。对于Central和Peripheral有多种说法,上面我们说的是主从,还有客户端/服务端,中心设备/外围设备,我们这里简单介绍一下,客户端(Client)对应上面的Central,接收数据;服务端(Server)对应上面的额Peripheral,提供数据,这个需要和网站的服务器/客户端区别一下;中心设备(Central)和外围设备(Peripheral),其实上面叫中心设备和外围设备。上面主设备(Master)和从设备(Slave)应该对应主/从。这个根据个人习惯,主/从用的比较多,如果在蓝牙中提到这些知道就行了。
1437 0
|
前端开发 Java 数据安全/隐私保护
聊聊 OAuth 2.0 的 Token 续期处理
Token 校验逻辑 // CheckTokenEndpoint.checkToken @RequestMapping(value = "/oauth/check_token") @ResponseBody public Map checkToken(@RequestPara.
2265 0
|
调度 iOS开发 MacOS
【MacOS 系列】mac常用快捷键收集,包含提高开发效率、精选快捷键、内置截图快捷键
【MacOS 系列】mac常用快捷键收集,包含提高开发效率、精选快捷键、内置截图快捷键
471 0
|
安全 Linux 测试技术
|
JavaScript Java 应用服务中间件
使用 Docker 高效搭建本地开发环境(详细教程)
使用 Docker 高效搭建本地开发环境(详细教程)
15922 0
使用 Docker 高效搭建本地开发环境(详细教程)
|
1月前
|
Web App开发 缓存 监控
内存溢出与内存泄漏:解析与解决方案
本文深入解析内存溢出与内存泄漏的区别及成因,结合Java代码示例展示典型问题场景,剖析静态集合滥用、资源未释放等常见原因,并提供使用分析工具、优化内存配置、分批处理数据等实用解决方案,助力提升程序稳定性与性能。
560 1
|
9月前
|
人工智能 IDE 开发工具
从0到1彻底掌握Trae:手把手带你实战开发AI Chatbot,提升开发效率的必备指南!
Trae是字节跳动推出的一款免费的AI集成的开发环境,集成了Claude3.5与GPT-4o等主流AI模型,提供AI问答、智能代码生成、智能代码补全,多模态输入等功能。支持界面全中文化,为中文开发者提供了高效的开发体验
5564 11
从0到1彻底掌握Trae:手把手带你实战开发AI Chatbot,提升开发效率的必备指南!
|
存储 监控 分布式数据库
百亿级存储架构: ElasticSearch+HBase 海量存储架构与实现
本文介绍了百亿级数据存储架构的设计与实现,重点探讨了ElasticSearch和HBase的结合使用。通过ElasticSearch实现快速检索,HBase实现海量数据存储,解决了大规模数据的高效存储与查询问题。文章详细讲解了数据统一接入、元数据管理、数据一致性及平台监控等关键模块的设计思路和技术细节,帮助读者理解和掌握构建高性能数据存储系统的方法。
百亿级存储架构: ElasticSearch+HBase 海量存储架构与实现
|
XML 前端开发 Java
深入理解Spring EL表达式的高级功能
深入理解Spring EL表达式的高级功能
915 1
|
存储 Linux Docker
在Docker中,本地的镜像文件都存放在哪里?
在Docker中,本地的镜像文件都存放在哪里?