当前位置: 首页 >前端技术 > 把axios封装为vue插件使用

把axios封装为vue插件使用

前言

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

封装的基本要求

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

文件结构

在src目录下新建 http 文件夹

把axios封装为vue插件使用 _ JavaClub全栈架构师技术笔记

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) {  retu 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. 带上tokenif (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)  }}retu config  },  error => {// 请求错误时console.log('request:', error)// 1. 判断请求超时if (error.code === 'ECONNABORTED' && error.message.indexOf('timeout') !== -1) {  console.log('timeout请求超时')  // retu service.request(originalRequest);//再重复请求一次}// 2. 需要重定向到错误页面const errorInfo = error.responseconsole.log(errorInfo)if (errorInfo) {  //error =errorInfo.data  //页面那边catch的时候就能拿到详细的错误信息,看最下边的Promise.reject  const errorStatus = errorInfo.status; // 404 403 500 ...  router.push({path: `/error/${errorStatus}`  })}retu 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 errretu data  },  err => {if (err && err.response) {  switch (err.response.status) {case 400:  err.message = '请求错误'  breakcase 401:  err.message = '未授权,请登录'  breakcase 403:  err.message = '拒绝访问'  breakcase 404:  err.message = `请求地址出错: ${err.response.config.url}`  breakcase 408:  err.message = '请求超时'  breakcase 500:  err.message = '服务器内部错误'  breakcase 501:  err.message = '服务未实现'  breakcase 502:  err.message = '网关错误'  breakcase 503:  err.message = '服务不可用'  breakcase 504:  err.message = '网关超时'  breakcase 505:  err.message = 'HTTP版本不受支持'  breakdefault:  }}console.error(err)retu Promise.reject(err) // 返回接口返回的错误信息  })// 请求处理instance(options).then(res => {  resolve(res)  retu false}).catch(error => {  reject(error)})  })}

interface.js

import axios from './api'/* 将所有接口统一起来便于维护 * 如果项目很大可以将 url 独立成文件,接口分成不同的模块 */// 单独导出export const query = () => {retu axios({url: '/query',method: 'get'})}  export const list = (id) => {retu axios({url: `/list${id}`,method: 'get'})}export const upload = data => {retu axios({url: '/upload',method: 'post',data})}// 默认全部导出export default {query,list,upload}

index.js

封装成 Vue 插件

// 导入所有接口import apiList from './interface'const install = Vue => {if (install.installed) retu;install.installed = true;Object.defineProperties(Vue.prototype, {// 注意哦,此处挂载在 Vue 原型的 $api 对象上$api: {get() {retu 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


参考文章

作者:_marven
来源链接:https://www.cnblogs.com/zhouyangla/p/9121779.html

版权声明:
1、JavaClub(https://www.javaclub.cn)以学习交流为目的,由作者投稿、网友推荐和小编整理收藏优秀的IT技术及相关内容,包括但不限于文字、图片、音频、视频、软件、程序等,其均来自互联网,本站不享有版权,版权归原作者所有。

2、本站提供的内容仅用于个人学习、研究或欣赏,以及其他非商业性或非盈利性用途,但同时应遵守著作权法及其他相关法律的规定,不得侵犯相关权利人及本网站的合法权利。
3、本网站内容原作者如不愿意在本网站刊登内容,请及时通知本站(javaclubcn@163.com),我们将第一时间核实后及时予以删除。





本文链接:https://www.javaclub.cn/front/68962.html

标签:vue iOS
分享给朋友:

“把axios封装为vue插件使用” 的相关文章

iOS中的物理引擎 2022年06月10日 21:23:55
iOS开发调试概览 2022年06月11日 16:02:02
iOS多线程GCD 2022年06月11日 19:00:59
iOS-汉字排序 2022年06月12日 18:30:14
iOS开发-UI 从入门到精通(四) 2022年06月16日 15:52:02
iOS开发tips-神奇的UITableView 2022年06月19日 09:31:47