目录
- 简易封装
- status.js
- interceptor.js
- http.js
- API接口文件
- 主代码
简易封装
在介绍如何封装axios之前,请大家先了解一下我们之后将要创建的三个文件及其含义,为后续理解代码做准备
三个主要文件:
status.js 保存状态码以及对应默认返回信息
interceptor.js 定义axios拦截器与初始url
http.js 封装GET POST请求
api-xxx.js 专门用于处理某项请求的API接口
以上四个文件在接下来的演示中都在同一个文件夹下!
status.js
在这里简单的定义几个装填码,后续我们会直接调用他们
代码清单 status.js
// 在这里存储所有状态码及其对应的所有信息
export const STATUS = {
success: {
id: 200,
msg: "请求成功",
},
failed: {
id: 400,
msg: "请求失败",
},
};
interceptor.js
这里需要安装三个库:axios、qs、element-plus
该拦截器包含了以下功能:
前端请求后端时自动从自身的localstorage中提取token,放置于请求头中发送给后端;
可以对GET请求的参数进行字符串化;
可以拦截响应体,并判断token是否过期而让用户重新登陆(重定向到登录页面);
代码清单:interceptor.js
import axios from "axios";
import qs from "qs";
import { ElMessage } from "element-plus";
import { STATUS } from "./status";
// 初始化后端服务器数据
const server = axios.create({
baseURL: "http://localhost:10001",
timeout: 30 * 1000,
});
// 请求拦截器
server.interceptors.request.use(
(config) => {
// 如果请求方法为GET,则字符串化params
if (config.method === "get") {
config.paramsSerializer = (params) => {
return qs.stringify(params, {
arrayFormat: "comma",
});
};
}
// 获取用户token并附带在headers里传递给后端
const token = localStorage.token;
if (token) {
config.headers["token"] = token;
}
return config;
},
(err) => {
Promise.reject(err);
}
);
// 响应拦截器
server.interceptors.response.use(
// 如果没错误就直接通过不拦截
(config) => {
return config;
},
(err) => {
// 如果错误的响应体存在的话
if (err.response) {
// 获取响应值message
const msg =
err.response.data === null
? "服务端出错,无法获取响应内容"
: err.response.data.message;
// 根据响应状态码来返回对应信息
switch (err.response.status) {
case STATUS.success.id:
ElMessage(STATUS.success.msg);
break;
default:
// 后端验证token无效,写到响应体里返回给前端
// 前端在此判定为无效,提示对应信息
if (msg === "invalid_token") {
ElMessage("token无效,请重新登陆");
} else {
ElMessage(msg);
}
break;
}
}
return Promise.reject(err);
}
);
// 导出拦截器供后续使用
export default server;
http.js
主要对 get 和 post 方法进行封装,设置 config,并判断有无 params 以及 formdata 来决定是否添加对应参数到请求体中
特别的,对于 post 请求的 config,必须要设置 header
中的 content-type
,否则后端无法识别传递过来的 formdata(比如 GIN 框架下就是无法识别的!)
代码清单:http.js
import qs from "qs";
import intcp from "./interceptor";
const http = {
// axios执行get请求
get(url, params) {
const config = {
method: "get",
url: url,
};
// 当存在params时,才会将其添加到config并发送给后端
if (params) config.params = params;
return intcp(config);
},
// axios执行post请求
post(url, formdata) {
const config = {
method: "post",
url: url,
// 这一条必须要加,否则后端无法识别formdata
headers: {
"Content-type": "application/x-www-form-urlencoded",
},
};
if (formdata) config.data = formdata;
console.log(config);
return intcp(config);
},
};
export default http;
API接口文件
下面仅展示前端接口文件书写,后端内容不是关键点,这里略过
这里我们编写一个用户CRUD接口,文件名为 api-modify.js
这里只是简单调用http进行二次封装,最终promise请求还需要在主代码中处理
为什么不直接在api中处理前后端交互逻辑的原因是:很多情况下我们会用到后端发回来的响应体中的值,写在主代码中可以直接获取它并变成响应式, 而直接在api里面写就会涉及到作用域以及数据传递问题,极不方便!!!
import http from "./http.js";
// 随意订制三个接口,明确各自的请求方法
export default {
selectUserList: (url) => {
return http.get(url)
},
insertUser: (url, data) => {
return http.post(url, data)
},
deleteUser: (url, data) => {
return http.post(url, data)
}
}
主代码
所谓主代码,即调用api接口的代码,在这里需要对api的promise进行then和catch,并针对性的做出响应
下方代码演示了,使用pinia配合api,获取后端值并存储的完整流程
import {onMounted, reactive, ref} from "vue";
import apiQuery from "../../api/api-query.js";
import dbStore from "../../store/db-store.js";
// 实例化store
const store = dbStore()
// 动态绑定store中数据,以便其一发生改变(前端取回后端返回数据),立即可得到渲染更新
let datas = reactive(store.$state.userLists)
// api接口,需自行处理then以及catch逻辑!
apiQuery.userDataQuery("/sdb/allusers").then(res => {
store.setUsersList(res)
console.log(res[0], "")
})