vue项目之axios

安装

yarn add --save axios vue-axios //不确定要不要
yarn add axios

axios 目录结构


image.png

index.ts

import router from "@/router";
import store from "@/store";
import axios, { AxiosRequestConfig, Method } from "axios";

// 定义接口
interface PendingType {
  url?: string;
  method?: Method;
  params: any;
  data: any;
  cancel: any;
}
// 请求失败后的错误统一处理

const errorHandle = (status: number, other: string) => {
  switch (status) {
    case 302: console.log('接口重定向了!');
      break;
    case 400:
      console.log("发出的请求有错误,服务器没有进行新建或修改数据的操作==>" + status)
      break;
    // 401: 未登录
    // 未登录则跳转登录页面,并携带当前页面的路径
    // 在登录成功后返回当前页面,这一步需要在登录页操作。
    case 401: //重定向
      console.log("token:登录失效==>" + status + ":" + store.state.Roles)
      localStorage.remove(store.state.Roles)
      localStorage.get(store.state.Roles)
      router.replace({
        path: '/Login',
      });
      break;
    // 403 token过期
    // 清除token并跳转登录页
    case 403:
      console.log("登录过期,用户得到授权,但是访问是被禁止的==>" + status)
      // store.commit('token', null);
      setTimeout(() => {
        router.replace({
          path: '/Login',
        });
      }, 1000);
      break;
    case 404:
      console.log("网络请求不存在==>" + status)
      break;
    case 406:
      console.log("请求的格式不可得==>" + status)
      break;
    case 408: console.log(" 请求超时!")
      break;
    case 410:
      console.log("请求的资源被永久删除,且不会再得到的==>" + status)
      break;
    case 422:
      console.log("当创建一个对象时,发生一个验证错误==>" + status)
      break;
    case 500:
      console.log("服务器发生错误,请检查服务器==>" + status)
      break;
    case 502:
      console.log("网关错误==>" + status)
      break;
    case 503:
      console.log("服务不可用,服务器暂时过载或维护==>" + status)
      break;
    case 504:
      console.log("网关超时==>" + status)
      break;
    default:
      console.log("其他错误错误==>" + status)
  }
}

// 取消重复请求
const pending: Array<PendingType> = []
const CancelToken = axios.CancelToken

// 移除重复请求
const removePending = (config: AxiosRequestConfig) => {
  for (const key in pending) {
    const item: number =+ key;
    const list: PendingType = pending[key];
    // 当前请求在数组中存在时执行函数体
    if(list.url === config.url && list.method === config.method && JSON.stringify(list.params) === JSON.stringify(config.params) && JSON.stringify(list.data) === JSON.stringify(config.data)){
      // 执行取消操作
      list.cancel('操作太频繁,请稍后再试')
      // 从数组中移除记录
      pending.splice(item, 1)
    }
  }
}

// 1.实例化请求配置
const instance = axios.create({
  baseURL: process.env.VUE_APP_API_URL,
  timeout: 3000,
  headers: {
    //php 的 post 传输请求头一定要这个 不然报错 接收不到值
    "Content-Type": "application/json;charset=UTF-8",
    "Access-Control-Allow-Origin-Type": '*'
  }
})
// 2.请求拦截器
instance.interceptors.request.use((config) => {
  // 在请求之前做什么
  console.log('interceptors request : ', config)
  removePending(config);
  config.cancelToken = new CancelToken((c) => {
    pending.push({ url: config.url, method: config.method, params: config.params, data: config.data, cancel: c });
  })
  // if(localStorage.get(store.state.Roles)) {
  //   config.headers.Authorization = localStorage.get(store.state.Roles);
  // }
  return config
}, (error) => {
  // 请求错误时做些什么~
  return Promise.reject(error)
})

// 3.响应拦截器
instance.interceptors.response.use((response)=>{
  // 对响应的数据做什么~
  console.log('response : ', response)
  removePending(response.config)
  // 请求成功
  if(response.status === 200 || response.status === 204) {
    return Promise.resolve(response.data.data)
  }else{
    return Promise.reject(response)
  }
  // return response
},(error)=>{
  const { response } = error;
  if (response) {
    errorHandle(response.status, response.data.message);

    // 超时重新请求
    const config = error.config;
    // 全局的请求次数,请求的间隙
    const [RETRY_COUNT, RETRY_DELAY] = [3, 1000];

    if (config && RETRY_COUNT) {
      // 设置用于跟踪重试计数的变量
      config.__retryCount = config.__retryCount || 0;
      // 检查是否已经把重试的总数用完
      if (config.__retryCount >= RETRY_COUNT) {
        return Promise.reject(response || { message: error.message });
      }
      // 增加重试计数
      config.__retryCount++;
      // 创造新的Promise来处理指数后退
      const backoff = new Promise<void>((resolve) => {
        setTimeout(() => {
          resolve();
        }, RETRY_DELAY || 1);
      });
      // instance重试请求的Promise
      return backoff.then(() => {
        return instance(config);
      });
    }

    return Promise.reject(response);
  } else {
    // 处理断网的情况
    // eg:请求超时或断网时,更新state的network状态
    // network状态在app.vue中控制着一个全局的断网提示组件的显示隐藏
    // 后续增加断网情况下做的一些操作
    // store.commit('networkState', false);
  }
  // return Promise.reject(error)
})

export default instance

request.ts

import axios from "."
// import qs from "qs";

// 主要用于封装基于axios配置的get/post/put/delete等使用方法。
export class request {
  static get = (url: string, params?: any) => {
    return new Promise((resolve, reject) => {
      axios.get(url, {params: params}).then((res: unknown) => {
        resolve(res)
      }).catch((err: any)=>{
        reject(err)
      })
    })
  }

  static post = (url: string, params?: any) => {
    return new Promise( (resolve, reject) => {
      axios.post(url, params).then( res =>{
        resolve(res)
      }).catch( err =>{
        reject(err)
      })
    })
  }
}

apis.ts

import { request } from "./request";

// 所有接口数据信息
class api {
  async login(loginInfo: any): Promise<any>{
    return (await request.post(`user/login`, loginInfo))
  }
}

export default new api()

使用

setup(){
    const store = useStore()
    const login = ()=>{
      const params = {
        code: "86",
        keep_alive: false,
        password: "qxia1213",
        phone: "18174956541",
      }
      store.dispatch('user/login', params)
    }
    return {
      login
    }
  }
// 调用
apis.login(data).then((user: User)=>{
   context.commit('SET_TOKEN', user.token || '')
   context.commit('SET_USER', user)
   resolve(user)
}).catch((error) => {
   context.commit('SET_TOKEN', '')
   context.commit('SET_USER', null)
   reject(error)
})
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 219,039评论 6 508
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,426评论 3 395
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 165,417评论 0 356
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,868评论 1 295
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,892评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,692评论 1 305
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,416评论 3 419
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,326评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,782评论 1 316
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,957评论 3 337
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,102评论 1 350
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,790评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,442评论 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,996评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,113评论 1 272
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,332评论 3 373
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,044评论 2 355

推荐阅读更多精彩内容