把axios封装为vue插件使用_Marven_的博客-程序员秘密

技术标签: vue  axios  

前言

自从Vue2.0推荐大家使用 axios 开始,axios 被越来越多的人所了解。使用axios发起一个请求对大家来说是比较简单的事情,但是axios没有进行封装复用,项目越来越大,引起的代码冗余。就会非常麻烦的一件事。所以本文会详细的跟大家介绍,如何封装请求,并且在项目组件中复用请求。有需要的朋友可以做一下参考。

封装的基本要求

  • 统一 url 配置
  • 统一 api 请求
  • request (请求)拦截器,例如:带上token等,设置请求头
  • response (响应)拦截器,例如:统一错误处理,页面重定向等
  • 根据需要,结合 Vuex 做全局的loading动画,或者错误处理
  • 将 axios 封装成 Vue 插件使用

文件结构

在src目录下新建 http 文件夹

config.js axios的默认配置
api.js 二次封装axios,拦截器等
interface.js 请求接口文件
index.js 将axios封装成插件

config.js

完整配置请参考 axios 的官方文档

export default {
  method: 'get',
  // 基础url前缀
  baseURL: 'https://www.example.com/api',
  // 请求头信息
  headers: {
    'Content-Type': 'application/json;charset=UTF-8'
  },
  // 参数
  data: {},
  // 设置超时时间
  timeout: 10000,
  // 携带凭证
  withCredentials: true,
  // 返回数据类型
  responseType: 'json'
}

api.js

import axios from 'axios';
import config from './config';
import qs from 'qs';
import Cookies from "js-cookie";
import router from '@/router'

// 使用vuex做全局loading时使用
// import store from '@/store'

export default function $axios(options) {
  return new Promise((resolve, reject) => {
    const instance = axios.create({
      baseURL: config.baseURL,
      headers: {},
      transformResponse: [function (data) {
      }]
    })

    // request 拦截器
    instance.interceptors.request.use(
      config => {
        let token = Cookies.get('markToken')
        // 1. 请求开始的时候可以结合 vuex 开启全屏 loading 动画
        // console.log(store.state.loading)
        // console.log('准备发送请求...')
        // 2. 带上token
        if (token) {
          config.headers.accessToken = token
        } else {
          // 重定向到登录页面
          router.push('/login')
        }
        // 3. 根据请求方法,序列化传来的参数,根据后端需求是否序列化
        if (config.method === 'post') {
          if (config.data.__proto__ === FormData.prototype
            || config.url.endsWith('path')
            || config.url.endsWith('mark')
            || config.url.endsWith('patchs')
          ) {
          } else {
            config.data = qs.stringify(config.data)
          }
        }
        return config
      },

      error => {
        // 请求错误时
        console.log('request:', error)
        // 1. 判断请求超时
        if (error.code === 'ECONNABORTED' && error.message.indexOf('timeout') !== -1) {
          console.log('timeout请求超时')
          // return service.request(originalRequest);//再重复请求一次
        }
        // 2. 需要重定向到错误页面
        const errorInfo = error.response
        console.log(errorInfo)
        if (errorInfo) {
          //error =errorInfo.data  //页面那边catch的时候就能拿到详细的错误信息,看最下边的Promise.reject
          const errorStatus = errorInfo.status; // 404 403 500 ...
          router.push({
            path: `/error/${errorStatus}`
          })
        }
        return Promise.reject(error) // 在调用的那边可以拿到(catch)你想返回的错误信息
      }
    )

    // response 拦截器
    instance.interceptors.response.use(
      response => {
        let data;
        // IE9时response.data是undefined,因此需要使用response.request.responseText(Stringify后的字符串)
        if (response.data == undefined) {
          data = JSON.parse(response.request.responseText)
        } else {
          data = response.data
        }

        // 根据返回的code值来做不同的处理
        switch (data.rc) {
          case 1:
            console.log(data.desc)
            break;
          case 0:
            store.commit('changeState')
            // console.log('登录成功')
          default:
        }
        // 若不是正确的返回code,且已经登录,就抛出错误
        // const err = new Error(data.desc)
        // err.data = data
        // err.response = response
        // throw err


        return data
      },
      err => {
        if (err && err.response) {
          switch (err.response.status) {
            case 400:
              err.message = '请求错误'
              break

            case 401:
              err.message = '未授权,请登录'
              break

            case 403:
              err.message = '拒绝访问'
              break

            case 404:
              err.message = `请求地址出错: ${err.response.config.url}`
              break

            case 408:
              err.message = '请求超时'
              break

            case 500:
              err.message = '服务器内部错误'
              break

            case 501:
              err.message = '服务未实现'
              break

            case 502:
              err.message = '网关错误'
              break

            case 503:
              err.message = '服务不可用'
              break

            case 504:
              err.message = '网关超时'
              break

            case 505:
              err.message = 'HTTP版本不受支持'
              break

            default:
          }
        }
        console.error(err)
        return Promise.reject(err) // 返回接口返回的错误信息
      }
    )

    // 请求处理
    instance(options).then(res => {
      resolve(res)
      return false
    }).catch(error => {
      reject(error)
    })
  })
}

interface.js

import axios from './api'

/* 将所有接口统一起来便于维护
 * 如果项目很大可以将 url 独立成文件,接口分成不同的模块
 */

// 单独导出
export const query = () => {
    return axios({
        url: '/query',
        method: 'get'
    })
}

export const list = (id) => {
    return axios({
        url: `/list${id}`,
        method: 'get'
    })
}

export const upload = data => {
    return axios({
        url: '/upload',
        method: 'post',
        data
    })
}

// 默认全部导出
export default {
    query,
    list,
    upload
}

index.js

封装成 Vue 插件

// 导入所有接口
import apiList from './interface'
const install = Vue => {
    if (install.installed) 
        return;
    install.installed = true;

    Object.defineProperties(Vue.prototype, {
        // 注意哦,此处挂载在 Vue 原型的 $api 对象上
        $api: {
            get() {
                return apiList
            }
        }
    })
}

export default install

使用

到此为止,万事俱备就差用了,在 mian.js 中做如下操作:

import api from './http/index'
Vue.use(api)
// 此时可以直接在 Vue 原型上调用 $api 了

在 vue 中使用

// List.vue

...
this.$api.list(id).then(res => {
     if (res.rc === 0) {
          this.pageList = res.data.item
      } else {
        this.$Message.info(res.desc);
      }
     })
     .catch(error => {
        this.$Message.info(error);
      })
...

总结

  • 以上二次封装较为全面,基本完成了我们之前的需求
  • 在错误的处理上还需要与后端协定好返回值,做具体的约定

本文同步发布在 https://www.cssge.com


参考文章

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/qq_29824621/article/details/80537904

智能推荐

关于Bert模型参数的分布 简单的数学题,另外 M代表百万个_愚昧之山绝望之谷开悟之坡的博客-程序员秘密

参数分布 Bert模型的版本如下: BERT-Base, Uncased: 12-layer, 768-hidden, 12-heads, 110M parameters BERT-Large, Uncased: 24-layer, 1024-hidden, 16-heads, 340M parameters BERT-Base, Cased: 12-layer, 768-hidden, 12-heads , 110M parameters BERT-La...

USART中DMA使用介绍_usart dma_wupeipeng的博客-程序员秘密

1 初始化代码分析1.1 DMA初始化代码作用是设置对应DMA通道的中断优先级;void MX_DMA_Init(void){​ /* DMA controller clock enable */ __HAL_RCC_DMA1_CLK_ENABLE();​ /* DMA interrupt init */ /* DMA1_Channel4_IRQn interrupt configuration */ HAL_NVIC_SetPriority(DMA1_Channel...

Java 跨域问题解决_dyyshb的博客-程序员秘密

1、什么就算跨域了,以下三点中只要有一个不同就是跨域啦。1)、协议不同;2)、域名不同(IP地址不同)、3)、端口号不同下图来自百度百科2、测试案例有两个站点:1)、自定义服务站点(假设为A站点),2)、smland-portal站点(假设为B站点),A站点服务地址假设为http:A/;B站点服务地址假设为http:B/;B站点想访问A站点服务信息B站点前端js文件...

中国编程第一人,一人抵一城!_Ch97CKd的博客-程序员秘密

点击上方“码农突围”,马上关注这里是码农充电第一站,回复“666”,获取一份专属大礼包真爱,请设置“星标”或点个“在看”有这样一位年轻人。他获得过百度之星程序设计大赛冠军、谷歌编...

python 判断文本所用的语言_编写简单的Python程序来判断文本的语种_weixin_39702483的博客-程序员秘密

1.问题的描述用Python进行文本处理时,有时候处理的文本中包含中文、英文、日文等多个语系的文本,有时候不能同时进行处理,这个时候就需要判别当前文本是属于哪个语系的。Python中有个langid工具包提供了此功能,langid目前支持97种语言的检测,非常好用。2.程序的代码以下Python是调用langid工具包来对文本进行语言检测与判别的程序代码:import langid ...

Nginx流量拷贝模块—ngx_http_mirror_module分析_weixin_33779515的博客-程序员秘密

原文出处:https://blog.csdn.net/xhjcehust/article/details/77093074 本文适合对nginx比较感兴趣的同学阅读,需要具备一定的服务端编程知识。一、背景最近nginx官网公布了nginx1.13.4最新的ngx_http_mirror_module模块,利用mirror模块,业务可以将线上实时访问流量拷贝至其他环境,基于这些...

随便推点

VBS中字符串比较的理解_vbs 字段值对比_Dnight的博客-程序员秘密

由于在VBS里赋值和比较都是用=号所以比较时要特别小心,一个例子: rs为一个recordset对象rs("name")="lily" 这里其实是个赋值语句要比较的话,应给这样写: trim(rs("name"))="lily" 因为trim函数返回值是字符串

python课堂笔记手抄图片_最简单的读书手抄报图片大全_惟殷有典的博客-程序员秘密

在知识创新和尊重知识、尊重人才的社会环境下,社会进步、个人发展都离不开知识。为了民族振兴,国民应弘扬读书精神,大兴读书之风。同时要改变读书观念,更新读书方式,提高阅读效率。下面是学习啦小编为大家带来的最简单的图片大全,希望大家喜欢。最简单的读书手抄报图片欣赏最简单的读书手抄报图一最简单的读书手抄报图二最简单的读书手抄报图三最简单的读书手抄报图四最简单的读书手抄报图五最简单的读书手抄报资料:读书的乐...

数据产品经理训练营_liudada8265的博客-程序员秘密

数据产品经理今年特别火比较火,市场平均薪资已达到26K+,疫情影响下,其他岗位都在收缩,唯数据产品经理独树一帜,不但高薪,还大量扩招。数据产品经理的市场需求在增加数据产品经理火的原因之一...

使用 AppFuse 的七个理由_Mountain_的博客-程序员秘密

开始学习在 Java 平台上使用诸如 Spring、Hibernate 或 MySQL 之类的开放源码工具时可能非常困难。再加上 Ant 或 Maven,以及与 DWR 一起的小 Ajax,还有 Web 框架 —— 即 JSF,我们必须睁大眼睛盯着如何配置应用程序。AppFuse 减少了集成开放源码项目的痛苦。它可以把测试变成一等公民,让我们可以从数据库表生成整个 UI,并使用 XFire 来支...

jQuery.FlexiGrid使用总结_iteye_21255的博客-程序员秘密

经过对FlexiGrid的大量使用,及时不时琢磨下其代码,对她的脾性有了一定的了解,是该做总结的时候了。 一、FlexiGrid下载1、原版代码最近Paulo P. Marinas对FlexiGrid做了修改,以适应jQuery 1.4.2。看了下作者列出来的特性表,该表很长,但与过去比,却没有增加什么新功能,所以版本号不做修改,仍然是1.0b3。其中提到的一点值得注意...

推荐文章

热门文章

相关标签