开箱即用的axios Api请求封装

之前做项目的时候经常遇到个问题:相同url相同参数的接口,短时间内重复请求调用。
产生的原因有多种:
按钮未做防抖/节流/锁,高频点击(实际上项目中所有按钮都加上防抖/节流/锁处理是不现实的);
相同信息同页面的异构展示,每个地方独立获取数据(不够优雅的架构等);
代码没优化好,但是历史代码逻辑复杂,不敢重构......
最开始想要做的优化方案是:对每一个请求做记录(通过url和param),请求发起前检测是否存在已发出的接口,如果有则打断。但是有一个问题,被打断的接口没有返回,不符合一些特殊的业务场景。
于是在原有的axios封装上不断优化,历时两天,终于撸出最终版:


image.png

目录结构如图。
首先封装工具函数:requestMeans.js


import qs from 'qs'; //参数编译
import { getToken } from '@/storage';
 
let pending = []; //用于存储每个ajax请求的取消函数和ajax标识
let task = {}; //用于存储每个ajax请求的处理函数,通过请求结果调用,以ajax标识为key
 
//请求开始前推入pending
export const pushPending = (item) => {
  pending.push(item);
};
//请求完成后取消该请求,从列表删除
export const removePending = (e) => {
  let key = e.baseURL + '?' + (e.method == 'post' ? e.data : qs.stringify(e.params));
  for (let p in pending) {
    if (pending[p].key === key) {
      //当前请求在列表中存在时
      pending[p].cancelToken(); //执行取消操作
      pending.splice(p, 1); //把这条记录从列表中移除
    }
  }
};
 
// 创建task
export const createTask = (key, resolve) => {
  let callback = (response) => {
    if (response.data.status === -1) {
      // 这里处理授权逻辑
    } else if (response.data.status) {
      // 这里做全局错误提示
    }
    resolve(response.data);
  };
  if (!task[key]) task[key] = [];
  task[key].push(callback);
};
// 处理task
export const handleTask = (key, response) => {
  for (let i = 0; task[key] && i < task[key].length; i++) {
    task[key][i](response);
  }
  task[key] = undefined;
};
 
// data处理:存在token则加入
export const dataAddToken = (data) => {
  const token = getToken();
  if (token) {
    data['token'] = token;
  }
  return data;
};

请求封装:request.js


import qs from 'qs'; //参数编译
import axios from 'axios';
 
import { pushPending, removePending, createTask, handleTask, dataAddToken } from './requestMeans';
 
const getHeaders = { 'Content-Type': 'application/json' };
const postHeaders = { 'Content-Type': 'application/x-www-form-urlencoded' };
const fileHeaders = { 'Content-Type': 'multipart/form-data' };
 
const baseURL = 'https://mock.apipost.cn/app/mock/project/f0a1c097-861b-473f-b0d4-a706fbe0af91';
 
//请求封装
export const request = (method, url, params, headers, preventRepeat = true, uploadFile = false) => {
  let key = baseURL + url + '?' + qs.stringify(params);
  return new Promise((resolve, reject) => {
    const instance = axios.create({
      baseURL: baseURL + url,
      headers,
      timeout: 30 * 1000,
    });
 
    instance.interceptors.request.use(
      (config) => {
        if (preventRepeat) {
          removePending(config); //防止多次触发请求
          config.cancelToken = new axios.CancelToken((cancelToken) => {
            pushPending({ key, cancelToken });
          });
        }
        return config;
      },
      (err) => {
        return Promise.reject(err);
      }
    );
 
    instance.interceptors.response.use(
      (response) => {
        if (preventRepeat) {
          removePending(response.config);
        }
        return response;
      },
      (error) => {
        return Promise.reject(error);
      }
    );
 
    // 请求执行前加入task
    createTask(key, resolve);
 
    instance(Object.assign({}, { method }, method === 'post' || method === 'put' ? { data: !uploadFile ? qs.stringify(params) : params } : { params }))
      .then((response) => {
        // 处理task
        handleTask(key, response);
      })
      .catch();
  });
};
 
// 定义对外Get、Post请求
export default {
  // 单独导出 用于put等非常规请求及需要特殊处理header的请求
  request,
  get(url, data = {}, preventRepeat = true) {
    data = dataAddToken(data);
    return request('get', url, data, getHeaders, preventRepeat, false);
  },
  post(url, data = {}, preventRepeat = true) {
    data = dataAddToken(data);
    return request('post', url, data, postHeaders, preventRepeat, false);
  },
  file(url, data = {}, preventRepeat = true) {
    return request('post', url, data, fileHeaders, preventRepeat, true);
  },
};

api封装:index.js


import Request from './request';
 
// 公用-获取验证码
export const postMerchantLoginCms = (data) => Request.post('/merchant/login/cms', data);
 
// 临时mock接口,用于测试封装功能
// export const getList = (data) => Request.get('/web-model-library/merchant/investigation', data);
export const getList = (data) => Request.post('/active/getRandModel', data);

使用方式:

import { getList } from "@/api";

 methods: {
     async getdata() {
      let data =await getList({a:1});
    },
}

作者:断律绎殇
链接:https://juejin.cn/post/7087787561447325732
来源:稀土掘金
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

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

推荐阅读更多精彩内容