前端token知识梳理:token如何存储?token过期如何处理?如何无感刷新token?

本文探讨了在前后端交互中token的存储方式,包括cookie和localStorage,并详细阐述了token过期时的处理方法,如重新登录或刷新token。通过使用访问令牌和刷新令牌,可以在token过期时无痛刷新,保持用户登录状态。同时,文章介绍了在axios拦截器中实现token刷新的逻辑,确保用户体验的连续性。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

在前后端是以token的形式交互,既然是token,那么肯定有它的过期时间(为了接口数据的安全,服务器的token一般不会设置太长,根据需要一般是1-7天的样子),没有一个token是永久的,永久的token就相当于一串永久的密码,是不安全的。

登录场景

在这里插入图片描述

那么既然token既然有过期时间,问题就来了

一、前后端交互的过程中token如何存储?

方法1:存在 cookie 中

cookie的大小约4k,兼容性在ie6及以上 都兼容,在浏览器和服务器间来回传递,因此它得在服务器的环境下运行,而且可以设定过期时间,默认的过期时间是session会话结束。

方法2:存在 localStorage 中

localStorage的大小约5M,兼容性在ie7及以上都兼容,有浏览器就可以,不需要在服务器的环境下运行, 会一直存在,除非手动清除 。

二、token过期时,前端该怎么处理?

方法1:跳回登陆页面重新登陆(不推荐)

在vue中我们可以在 axios 拦截器中这样写:

instance.interceptors.response.use(
  function (response) {
    // 对响应数据做点什么
    return response.data
  },
  function (error) {
    if (error.response) {
      if (error.response.status === 401) {
            Message.error('登陆过期请重新登陆!')
            setToken('')
            router.push({
              name: 'login'
            })
        }
      }
    }
    // 对响应错误做点什么
    return Promise.reject(error.response)
  }
)

方法2:重新获取token

出于本文的目的,我们将关注两种最常见的令牌类型:访问令牌和刷新令牌。

  • 访问令牌携带必要的信息以直接访问资源。换句话说,当客户端将访问令牌传递给管理资源的服务器时,该服务器可以使用令牌中包含的信息来决定客户端是否被授权。访问令牌通常具有到期日期并且是短暂的。
    在这里插入图片描述

  • 刷新令牌包含获取新访问令牌所需的信息。换句话说,每当访问令牌需要访问特定资源时,客户端可以使用刷新令牌来获得由认证服务器发布的新访问令牌。常见用例包括在旧的访问令牌过期后获取新访问令牌,或者首次访问新资源。刷新令牌也可以过期,但相当长寿。刷新令牌通常受到严格的存储要求,以确保它们不会泄露。
    在这里插入图片描述

总结: 服务器生成token的过程中,会有两个时间,一个是token失效时间,一个是token刷新时间。
刷新时间肯定比失效时间长,当用户的 token 过期时,你可以拿着过期的token去换取新的token,来保持用户的登陆状态,当然你这个过期token的过期时间必须在刷新时间之内,如果超出了刷新时间,那么返回的依旧是 401

所以要实现无痛刷新token,我们应该这样

  1. 在axios的拦截器中加入token刷新逻辑
  2. 当用户token过期时,去向服务器请求新的 token
  3. 把旧的token替换为新的token
  4. 然后继续用户当前的请求

在axios的拦截器中加入token刷新逻辑:

instance.interceptors.response.use(
  function (response) {
    // 对响应数据做点什么
    return response.data
  },
  function (error) {
    if (error.response) {
      if (error.response.status === 401) {
        // 如果当前路由不是login,并且用户有 “记住密码” 的操作
        // 那么去请求新 token
        if (router.currentRoute.name !== 'login') {
          if (getRemember() && getRefreshToken()) {
            return doRequest(error)
          } else {
            Message.error('登陆过期请重新登陆!')
            setToken('')
            router.push({
              name: 'login'
            })
          }
        }
      }
    }
    // 对响应错误做点什么
    return Promise.reject(error.response)
  }
)

async function doRequest (error) {
  const data = await store.dispatch('refreshToken')
  let { token_type: tokenType, access_token: accessToken } = data
  let token = tokenType + accessToken
  let config = error.response.config
  config.headers.Authorization = token
  const res = await axios.request(config)
  return res
}
 
// refreshToken 中重新设置了 token 和  refresh_token
commit('setToken', { token, expiresIn })
setRefreshToken(token, refreshTtl / (60 * 60 * 24))

总结

弄两个token,一个负责鉴权得token:access_token,一个负责刷新得token:refresh_token,
每次请求的时候都带上这两个token,后端拦截器判断,先判断鉴权access_token是否有效和过期,如果有效的话,就允许访问。如果过期了,就判断刷新refresh_token是否有效,如果有效,就返回指定状态码,然后让前端根据这个状态码去吊用刷新token接口。如果刷新token失效了,就提示需要重新登录!

https://juejin.cn/post/6854573219119104014#heading-9

在React应用中,从Cookie中获取Token通常需要进行以下几个步骤: 1. **浏览器访问**: 首先,当用户登录时,服务器会将Token设置为Cookie,并返回给客户端(通常是前端)。React组件作为用户的界面层,并不会直接操作Cookie,它依赖于浏览器的JavaScript API。 2. **JavaScript API**: 使用JavaScript的`document.cookie`属性可以读取所有的Cookie。例如,你可以这样做: ```javascript function getTokenFromCookie() { const cookies = document.cookie.split(';'); for (let i = 0; i < cookies.length; i++) { const cookiePair = cookies[i].trim().split('='); if (cookiePair[0] === 'your_cookie_name') { // 替换为实际的cookie名 return decodeURIComponent(cookiePair[1]); // 解码Token值 } } return null; } ``` 3. **状态管理**: 获取到Token后,通常需要将其保存在一个应用程序的状态管理库(如Redux、MobX或React Context)里,以便在整个应用生命周期内使用。例如,在Redux中: ```javascript import { useSelector } from 'react-redux'; function useToken() { const token = useSelector(state => state.token); // 如果未找到Token,可以从Cookie获取 return token || getTokenFromCookie(); } // 使用这个hook的地方 const App = () => { const token = useToken(); //... }; ``` 4. **安全性注意事项**: 一定要注意,尽管Cookie有时用于存储敏感信息,但它们不是安全的存储手段,因为它们是明文传输的。如果Token包含敏感信息,最好使用更安全的方式,如本地存储或JWT(JSON Web Tokens)。 **相关问题:** 1. 怎么处理过期的Cookie中的Token? 2. 使用React如何避免跨域访问时Cookie的问题? 3. Redux之外,还有哪些状态管理方案可以结合使用获取和存储Token?
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

卸载引擎

如果对您有帮助,请赏个饭吃

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

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

打赏作者

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

抵扣说明:

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

余额充值