把axios封装为vue插件使用
前言
自从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) { 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),我们将第一时间核实后及时予以删除。