axios 的封装 在 vue 项目中,和后台交互获取数据这块,我们通常使用的是 axios 库,它是基于 promise 的 http 库,可运行在浏览器端和 node.js 中。他有很多优秀的特性,例如拦截请求和响应、取消请求、转换 json、客户端防御 XSRF 等。所以我们的尤大大也是果断放弃了对其官方库vue-resource 的维护,直接推荐我们使用 axios 库。如果还对 axios 不了解的,可以移步axios 文档 。
安装 引入 在项目的 src 目录中,新建一个 request 文件夹,然后在里面新建一个 http.js 和一个 api.js 文件。http.js 文件用来封装我们的 axios,api.js 用来统一管理我们的接口。
1 2 3 4 5 import axios from "axios" ; import QS from "qs" ; import { Message } from "element-ui" ;
环境的切换 项目环境可能有开发环境、测试环境和生产环境。需要通过 node 的环境变量来匹配默认的接口 url 前缀。axios.defaults.baseURL 可以设置 axios 的默认请求地址。
1 2 3 4 5 6 7 8 9 10 11 if (process.env.NODE_ENV == "development" ) { axios.defaults.baseURL = "https://www.baidu.com" ; } else if (process.env.NODE_ENV == "test" ) { axios.defaults.baseURL = "https://www.ceshi.com" ; } else if (process.env.NODE_ENV == "production" ) { axios.defaults.baseURL = "https://www.production.com" ; }
设置请求超时 通过 axios.defaults.timeout 设置默认的请求超时时间。例如超过了 5s,就会告知用户当前请求超时,请刷新等。
1 axios.defaults.timeout = 5000 ;
post 请求头的设置 post 请求的时候,我们需要加上一个请求头,所以可以在这里进行一个默认的设置,即设置 post 的请求头为 application/x-www-form-urlencoded;charset=UTF-8
1 2 axios.defaults.headers.post["Content-Type" ] = "application/x-www-form-urlencoded;charset=UTF-8" ;
请求拦截器 我们在发送请求前可以进行一个请求的拦截,为什么要拦截呢,我们拦截请求是用来做什么的呢?比如,有些请求是需要用户登录之后才能访问的,或者 post 请求的时候,我们需要序列化我们提交的数据。这时候,我们可以在请求被发送之前进行一个拦截,从而进行我们想要的操作。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 import store from "@/store/index" ;axios.interceptors.request.use( (config) => { const token = store.state.token; token && (config.headers.Authorization = token); return config; }, (error) => { return Promise .error(error); } );
这里说一下 token,一般是在登录完成之后,将用户的 token 通过 localStorage 或者 cookie 存在本地,然后用户每次在进入页面的时候(即在 main.js 中),会首先从本地存储中读取 token,如果 token 存在说明用户已经登陆过,则更新 vuex 中的 token 状态。然后,在每次请求接口的时候,都会在请求的 header 中携带 token,后台人员就可以根据你携带的 token 来判断你的登录是否过期,如果没有携带,则说明没有登录过。
响应拦截器 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 axios.interceptors.response.use( response => { if (response.status === 200 ) { return Promise .resolve(response); } else { return Promise .reject(response); } }, error => { if (error.response.status) { switch (error.response.status) { case 401 : router.replace({ path: '/login' , query: { redirect: router.currentRoute.fullPath } }); break ; case 403 : Message({ message: '登录过期,请重新登录' , type:'error' , duration: 1000 , }); localStorage .removeItem('token' ); store.commit('loginSuccess' , null ); setTimeout (() => { router.replace({ path: '/login' , query: { redirect: router.currentRoute.fullPath } }); }, 1000 ); break ; case 404 : Message({ message: '网络请求不存在' , type:'error' , duration: 1500 , }); break ; default : Message({ message: error.response.data.message, type: 'error' , duration: 1500 }); } return Promise .reject(error.response); } } });
响应拦截器很好理解,就是服务器返回给我们的数据,我们在拿到之前可以对他进行一些处理。例如上面的思想:如果后台返回的状态码是 200,则正常返回数据,否则的根据错误的状态码类型进行一些我们需要的错误,其实这里主要就是进行了错误的统一处理和没登录或登录过期后调整登录页的一个操作。
封装 get 方法和 post 方法 我们常用的 ajax 请求方法有 get、post、put 等方法。axios 对应的也有很多类似的方法,不清楚的可以看下文档。但是为了简化我们的代码,我们还是要对其进行一个简单的封装。下面我们主要封装两个方法:get 和 post。
get 方法: 我们通过定义一个 get 函数,get 函数有两个参数,第一个参数表示我们要请求的 url 地址,第二个参数是我们要携带的请求参数。get 函数返回一个 promise 对象,当 axios 其请求成功时 resolve 服务器返回 值,请求失败时 reject 错误值。最后通过 export 抛出 get 函数。 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 export function get (url, params ) { return new Promise ((resolve, reject ) => { axios .get(url, { params: params, }) .then((res ) => { resolve(res.data); }) .catch((err ) => { reject(err.data); }); }); }
post 方法: 原理同 get 基本一样,但是要注意的是,post 方法必须要使用对提交从参数对象进行序列化的操作,所以这里我们通过 node 的 qs 模块来序列化我们的参数。这个很重要,如果没有序列化操作,后台是拿不到你提交的数据的。这就是文章开头我们 import QS from ‘qs’的原因。 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 export function post (url, params ) { return new Promise ((resolve, reject ) => { axios .post(url, QS.stringify(params)) .then((res ) => { resolve(res.data); }) .catch((err ) => { reject(err.data); }); }); }
这里有个小细节说下,axios.get()方法和 axios.post()在提交数据时参数的书写方式还是有区别的。区别就是,get 的第二个参数是一个{},然后这个对象的 params 属性值是一个参数对象的。而 post 的第二个参数就是一个参数对象。
api 接口统一管理 在 api.js 中引入封装的 get 和 post 方法
1 2 3 4 5 import { get, post } from "./http" ;export const Login = (data ) => post("api/v1/login" , data);
我们定义了一个 Login 方法,这个方法有一个参数 data,data 是我们请求接口时携带的参数对象。而后调用了我们封装的 post 方法,post 方法的第一个参数是我们的接口地址,第二个参数是 Login 的 data 参数,即请求接口时携带的参数对象。最后通过 export 导出 Login。
然后在我们的页面中可以这样调用我们的 api 接口:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 import { apiAddress } from "@/request/api" ; export default { name: "Address" , created ( ) { this .onLoad(); }, methods: { onLoad ( ) { Login({ userName: "admin" , password: 123456 , }).then((res ) => { }); }, }, };
api 接口管理的一个好处就是,我们把 api 统一集中起来,如果后期需要修改接口,我们就直接在 api.js 中找到对应的修改就好了,而不用去每一个页面查找我们的接口然后再修改会很麻烦。还有就是如果直接在我们的业务代码修改接口,一不小心还容易动到我们的业务代码造成不必要的麻烦。
完整 axios 封装代码 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 import axios from "axios" ;import QS from "qs" ;import { Message } from "element-ui" ;import store from "../store/index" ;if (process.env.NODE_ENV == "development" ) { axios.defaults.baseURL = "https://www.baidu.com" ; } else if (process.env.NODE_ENV == "test" ) { axios.defaults.baseURL = "https://www.ceshi.com" ; } else if (process.env.NODE_ENV == "production" ) { axios.defaults.baseURL = "https://www.production.com" ; } axios.defaults.timeout = 5000 ; axios.defaults.headers.post["Content-Type" ] = "application/x-www-form-urlencoded;charset=UTF-8" ; axios.interceptors.request.use( (config) => { const token = store.state.token; token && (config.headers.Authorization = token); return config; }, (error) => { return Promise .error(error); } ); axios.interceptors.response.use( (response) => { if (response.status === 200 ) { return Promise .resolve(response); } else { return Promise .reject(response); } }, (error) => { if (error.response.status) { switch (error.response.status) { case 401 : router.replace({ path: "/login" , query: { redirect : router.currentRoute.fullPath }, }); break ; case 403 : Message({ message: "登录过期,请重新登录" , type: "error" , duration: 1000 , }); localStorage .removeItem("token" ); store.commit("loginSuccess" , null ); setTimeout (() => { router.replace({ path: "/login" , query: { redirect: router.currentRoute.fullPath, }, }); }, 1000 ); break ; case 404 : Message({ message: "网络请求不存在" , type: "error" , duration: 1500 , }); break ; default : Message({ message: error.response.data.message, type: "error" , duration: 1500 , }); } return Promise .reject(error.response); } } ); export function get (url, params ) { return new Promise ((resolve, reject ) => { axios .get(url, { params: params, }) .then((res ) => { resolve(res.data); }) .catch((err ) => { reject(err.data); }); }); } export function post (url, params ) { return new Promise ((resolve, reject ) => { axios .post(url, QS.stringify(params)) .then((res ) => { resolve(res.data); }) .catch((err ) => { reject(err.data); }); }); }
根据需求不同作优化 优化 axios 封装,去掉之前的 get 和 post 断网情况处理 更加模块化的 api 管理 接口域名有多个的情况 api 挂载到 vue.prototype 上省去引入的步骤 http.js 中 axios 封装的优化,先直接贴代码:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 import axios from "axios" ;import router from "../router" ;import store from "../store/index" ;import { Message } from "element-ui" ;const tip = (msg ) => { Message({ message: msg, type: "error" , duration: 1000 , }); }; const toLogin = () => { router.replace({ path: "/login" , query: { redirect: router.currentRoute.fullPath, }, }); }; const errorHandle = (status, other ) => { switch (status) { case 401 : toLogin(); break ; case 403 : tip("登录过期,请重新登录" ); localStorage .removeItem("token" ); store.commit("loginSuccess" , null ); setTimeout (() => { toLogin(); }, 1000 ); break ; case 404 : tip("请求的资源不存在" ); break ; default : console .log(other); } }; var instance = axios.create({ timeout : 5000 });instance.defaults.headers.post["Content-Type" ] = "application/x-www-form-urlencoded" ; instance.interceptors.request.use( (config) => { const token = store.state.token; token && (config.headers.Authorization = token); return config; }, (error) => Promise .error(error) ); instance.interceptors.response.use( (res) => (res.status === 200 ? Promise .resolve(res) : Promise .reject(res)), (error) => { const { response } = error; if (response) { errorHandle(response.status, response.data.message); return Promise .reject(response); } else { if (!window .navigator.onLine) { store.commit("changeNetwork" , false ); } else { return Promise .reject(error); } } } ); export default instance;
如下几点改变:
去掉了之前 get 和 post 方法的封装,通过创建一个 axios 实例然后 export default 方法导出,这样使用起来更灵活一些。 去掉了通过环境变量控制 baseUrl 的值。考虑到接口会有多个不同域名的情况,所以准备通过 js 变量来控制接口域名。这点具体在 api 里会介绍。 增加了请求超时,即断网状态的处理。 当断网时,通过更新 vuex 中 network 的状态来控制断网提示组件的显示隐藏。断网提示一般会有重新加载数据的操作。 公用函数进行抽出,简化代码,尽量保证单一职责原则。 下面说下 api 这块,考虑到以下需求: 5. 更加模块化 6. 更方便多人开发,有效减少解决命名冲突 7. 处理接口域名有多个情况
这里新建了一个 api 文件夹,里面有一个 index.js 和一个 base.js,以及多个根据模块划分的接口 js 文件。index.js 是一个 api 的出口,base.js 管理接口域名,其他 js 则用来管理各个模块的接口。
先放 index.js 代码:
1 2 3 4 5 6 7 8 9 10 11 12 import article from "@/api/article" ;export default { article, };
index.js 是一个 api 接口的出口,这样就可以把 api 接口根据功能划分为多个模块,利于多人协作开发,比如一个人只负责一个模块的开发等,还能方便每个模块中接口的命名。
base.js:
1 2 3 4 5 6 7 8 9 const base = { sq: "https://xxxx111111.com/api/v1" , bd: "http://xxxxx22222.com/api" , }; export default base;
通过 base.js 来管理我们的接口域名,不管有多少个都可以通过这里进行接口的定义。即使修改起来,也是很方便的。 最后就是接口模块的说明,例如上面的 article.js:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 import base from "./base" ; import axios from "@/utils/http" ; import qs from "qs" ; const article = { articleList ( ) { return axios.get(`${base.sq} /topics` ); }, articleDetail (id, params ) { return axios.get(`${base.sq} /topic/${id} ` , { params: params, }); }, login (params ) { return axios.post(`${base.sq} /accesstoken` , qs.stringify(params)); }, }; export default article;
通过直接引入我们封装好的 axios 实例,然后定义接口、调用 axios 实例并返回,可以更灵活的使用 axios,比如你可以对 post 请求时提交的数据进行一个 qs 序列化的处理等。 请求的配置更灵活,你可以针对某个需求进行一个不同的配置。关于配置的优先级,axios 文档说的很清楚,这个顺序是:在 lib/defaults.js 找到的库的默认值,然后是实例的 defaults 属性,最后是请求的 config 参数。后者将优先于前者。 restful 风格的接口,也可以通过这种方式灵活的设置 api 接口地址。 最后,为了方便 api 的调用,我们需要将其挂载到 vue 的原型上。在 main.js 中:
1 2 3 4 5 6 7 import Vue from "vue" ;import App from "./App" ;import router from "./router" ; import store from "./store" ; import api from "./api" ; Vue.prototype.$api = api;
在页面中这样调用接口,例子 🌰:
1 2 3 4 5 6 7 8 9 methods: { onLoad (id ) { this .$api.article.articleDetail(id, { api: 123 }).then(res => { }) } }
关于断网的处理,这里做一个简单的示例 🌰:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 <template > <div id ="app" > <div v-if ="!network" > <h3 > 我没网了</h3 > <div @click ="onRefresh" > 刷新</div > </div > <router-view /> </div > </template > <script > import { mapState } from "vuex" ; export default { name: "App" , computed: { ...mapState(["network" ]), }, methods: { onRefresh ( ) { this .$router.replace("/refresh" ); }, }, }; </script >
这是 app.vue,这里简单演示一下断网。在 http.js 中介绍了,我们会在断网的时候,来更新 vue 中 network 的状态,那么这里我们根据 network 的状态来判断是否需要加载这个断网组件。断网情况下,加载断网组件,不加载对应页面的组件。当点击刷新的时候,我们通过跳转 refesh 页面然后立即返回的方式来实现重新获取数据的操作。因此我们需要新建一个 refresh.vue 页面,并在其 beforeRouteEnter 钩子中再返回当前页面。
1 2 3 4 5 6 beforeRouteEnter (to, from , next) { next(vm => { vm.$router.replace(from .fullPath) }) }