axios 获取 token 后重新请求失败的接口

本文介绍了一个使用axios的解决方案,用于在接口请求中遇到token过期问题时,自动刷新token并重新发起请求。通过创建一个特定的axios实例并在响应拦截器中处理token过期情况,实现了在多个子页面中无需重复编写业务代码即可自动处理token刷新的过程。

业务场景:有几个外部系统的接口,需要先通过一个获取 token 的接口获取到 token 后续业务接口需要在请求头中的 Authorization 带上 token。刚开始我在登录后获取了一次 token 并存储在 store 中,业务接口使用过的时候直接获取就可以了。后来发现这个 token 经常莫名其妙的过期,刚开始想着通过同步获取 token 后再请求业务接口,但是这些业务接口分散在多个子页面中,就会造成多写很多业务代码。所决定在这些外部接口使用一个特定的 axios 实例并在  response 的拦截处统一处理 token 问题

axios 实例代码:

import axios from 'axios'
import store from '@/store'
import { getTokenInfo } from '@/api/prophetApiMarket'
import { Message } from 'element-ui'
import debounce from 'lodash/debounce'

// 多个错误信息合并
const showMessage = debounce(function (message) {
  Message({
    message: message,
    type: 'error',
    duration: 3 * 1000
  })
}, 300)

// 最大重复次数
const MAXREPEAT = 3

// 记录当前接口重复请求次数,超过3次就不再请求了
let countState = {}

// 重新设置token
async function refreshToken(ajax, response) {
  // 获取token,data就是token字符串
  const { data } = await getTokenInfo()
  store.commit("setProphetToken", data)
  // 重新请求当前之前没有token 的接口
  response.config.headers['Authorization'] = "Bearer " + data
  ajax.defaults.headers.common['Authorization'] = "Bearer " + data
}

// 初始实例
const ajax = axios.create({
  timeout: 10000,
  baseURL: '/prop'
})

// 返回拦截
ajax.interceptors.response.use(async response => {
  if (response.status === 200) {
    const { data } = response
    // token过期
    if (data.code === 403 || (data.code === 406 && data.message === 'TOKEN已过期')) {
      countState[response.config.url] = countState[response.config.url] ?? 0
      // 最大重复次数
      if (countState[response.config.url] < MAXREPEAT) {
        // 计数
        countState[response.config.url]++
        // 获取token
        await refreshToken(ajax, response)
        // 重新请求
        return ajax(response.config)
      } else {
        // 重置计数,并返回结果
        countState[response.config.url] = 0
        return data
      }
    }

    return data
  } else {
    console.log('responseError: ' + response.data.message)
    showMessage(response.data.message)
    return Promise.resolve(response)
  }
}, error => {
  console.log('responseError: ' + error)
  showMessage(error.message)
})

export default ajax

引入上面文件使用:

import ajax from './request.js'
import store from '@/store'

// getToken
export const getTokenInfo = () => {
  return ajax.post('/aaaa', {
    secretKey: store.state.envVar.key,
    userId: store.state.envVar.id
  })
}
// getDataA
export const getDataA= (params) => {
  return ajax.post('/getDataA', params)
}

上面的 getDataA 是需要设置 Authorization 请求头的,当调用之后,response 拦截且匹配到 token 已过期就会先去获取 token 然后再重新请求

axios使用详见官网:axios中文文档|axios中文网

### Vue3 使用 Axios 获取 Token 的示例 在 Vue3 项目中使用 Axios 获取 Token 主要涉及配置 Axios 实例以及处理认证逻辑。以下是具体实现方式: #### 创建 Axios 实例 为了更好地管理 HTTP 请求,建议创建一个独立的 Axios 实例来设置默认参数和其他全局配置。 ```javascript // request/index.js import axios from 'axios'; const service = axios.create({ baseURL: process.env.VUE_APP_BASE_API, // API base_url timeout: 5000 // Request timeout }); export default service; ``` 此代码片段展示了如何初始化带有基础 URL 和超时时间的 Axios 客户端实例[^4]。 #### 登录接口定义 接下来定义用于登录操作的具体函数,该函数会向服务器发送用户名和密码,并接收返回的 Token 数据。 ```typescript // apis/user.ts import service from '@/request/index'; interface ILoginData { username: string, password: string } export function login(data: ILoginData) { return service({ url: '/login', // 替换成实际API路径 method: 'POST', data // 发送的数据体 }); } ``` 这段 TypeScript 代码描述了一个名为 `login` 的异步函数,它接受用户凭证作为输入并通过 POST 请求将其提交给指定的服务端点以换取访问令牌。 #### 存储与应用 Token 一旦成功获得 Token 后,则需妥善保存以便后续请求携带身份验证信息。通常做法是在本地存储 (localStorage 或 sessionStorage),并在每次发起请求前自动附加 Authorization 头部字段。 ```javascript // utils/auth.ts import { localStorage } from '@vueuse/core' function setToken(token:string){ window.localStorage.setItem('token', token); } function getToken(){ return window.localStorage.getItem('token'); } export {setToken,getToken}; ``` 上述 JavaScript 函数提供了两个工具方法:一个是用来把接收到的新 Token 放入浏览器缓存;另一个是从那里读取现有记录供其他组件调用[^1]. #### 自动附带 Token请求头 为了让所有发出的请求都包含有效的授权凭据,可以在 Axios 拦截器里做统一处理。 ```javascript // request/index.js 继续... service.interceptors.request.use( config => { const token = getToken(); if (token) { config.headers['Authorization'] = `Bearer ${token}`; } return config; }, error => Promise.reject(error) ); export default service; ``` 这里修改了之前创建好的 Axios 实例对象,在其请求拦截链上加入了一段逻辑,确保每一次对外通信都会带上当前用户的鉴权标志——即 Bearer 类型下的 Access Token[^3]. #### 调用 Login 并处理响应 最后一步就是在适当的地方触发这个流程,比如在一个表单提交事件处理器内部完成整个过程。 ```html <template> <!-- ... --> </template> <script setup lang="ts"> import { ref } from 'vue' import { useRouter } from 'vue-router' import { useStore } from '../store' import { login as loginUser } from '@/apis/user' import { setToken } from '@/utils/auth' const router = useRouter() const store = useStore() async function handleLogin() { try { let res = await loginUser({username:'admin',password:'123'}) console.log(res.data.token); // 打印出服务端返回的结果中的 token 字段 setToken(res.data.token); // 将获取到的 token 设置进 storage 中去 router.push('/dashboard'); // 成功后跳转至仪表盘页面或其他受保护路由 } catch(err){ alert(`登录失败:${err.message}`); // 如果发生错误则给出提示信息 } } </script> ``` 以上模板部分省略显,重点在于脚本标签内的业务逻辑编写。当点击按钮或者其他交互行为激活 `handleLogin()` 方法执行时,便会依照既定顺序依次尝试连接远程主机、解析 JSON 格式的回复内容并将其中的关键数据项提取出来加以利用[^2].
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值