一、开箱即用的axios封装:Vue3+ts
作者:诸葛小愚
链接:https://juejin.cn/post/7107047280133275678
来源:稀土掘金
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。
新建index.ts
文件:
- 需要定义请求返回的数据格式,这个可以和服务端约定好数据格式
- 需要定义axios的配置信息,用于在创建axios实例时传入
- 请求拦截器,前端所有的接口请求都会先达到请求拦截器,我们可以在此添加请求头信息
- 响应拦截器,服务端返回的数据会先达到响应拦截器,我们可以处理服务端的响应信息。如果是报错,就处理常见的报错;如果是成功,就返回数据
- 封装常用的get、put、post、delete接口方法
封装axios
import axios, {AxiosInstance, AxiosError, AxiosRequestConfig, AxiosResponse} from 'axios'
import {ElMessage} from 'element-plus'
// 数据返回的接口
// 定义请求响应参数,不含data
interface Result {
code: number;
msg: string
}
// 请求响应参数,包含data
interface ResultData<T = any> extends Result {
data?: T;
}
const URL: string = ''
enum RequestEnums {
TIMEOUT = 20000,
OVERDUE = 600, // 登录失效
FAIL = 999, // 请求失败
SUCCESS = 200, // 请求成功
}
const config = {
// 默认地址
baseURL: URL as string,
// 设置超时时间
timeout: RequestEnums.TIMEOUT as number,
// 跨域时候允许携带凭证
withCredentials: true
}
class RequestHttp {
// 定义成员变量并指定类型
service: AxiosInstance;
public constructor(config: AxiosRequestConfig) {
// 实例化axios
this.service = axios.create(config);
/**
* 请求拦截器
* 客户端发送请求 -> [请求拦截器] -> 服务器
* token校验(JWT) : 接受服务器返回的token,存储到vuex/pinia/本地储存当中
*/
this.service.interceptors.request.use(
(config: AxiosRequestConfig) => {
const token = localStorage.getItem('token') || '';
return {
...config,
headers: {
'x-access-token': token, // 请求头中携带token信息
}
}
},
(error: AxiosError) => {
// 请求报错
Promise.reject(error)
}
)
/**
* 响应拦截器
* 服务器换返回信息 -> [拦截统一处理] -> 客户端JS获取到信息
*/
this.service.interceptors.response.use(
(response: AxiosResponse) => {
const {data, config} = response; // 解构
if (data.code === RequestEnums.OVERDUE) {
// 登录信息失效,应跳转到登录页面,并清空本地的token
localStorage.setItem('token', '');
// router.replace({
// path: '/login'
// })
return Promise.reject(data);
}
// 全局错误信息拦截(防止下载文件得时候返回数据流,没有code,直接报错)
if (data.code && data.code !== RequestEnums.SUCCESS) {
ElMessage.error(data); // 此处也可以使用组件提示报错信息
return Promise.reject(data)
}
return data;
},
(error: AxiosError) => {
const {response} = error;
if (response) {
this.handleCode(response.status)
}
if (!window.navigator.onLine) {
ElMessage.error('网络连接失败');
// 可以跳转到错误页面,也可以不做操作
// return router.replace({
// path: '/404'
// });
}
}
)
}
handleCode(code: number):void {
switch(code) {
case 401:
ElMessage.error('登录失败,请重新登录');
break;
default:
ElMessage.error('请求失败');
break;
}
}
// 常用方法封装
get<T>(url: string, params?: object): Promise<ResultData<T>> {
return this.service.get(url, {params});
}
post<T>(url: string, params?: object): Promise<ResultData<T>> {
return this.service.post(url, params);
}
put<T>(url: string, params?: object): Promise<ResultData<T>> {
return this.service.put(url, params);
}
delete<T>(url: string, params?: object): Promise<ResultData<T>> {
return this.service.delete(url, {params});
}
}
// 导出一个实例对象
export default new RequestHttp(config);
实际使用
在使用时,我们需要在API文档中导入index.ts
,会自动创建一个axios实例。我们在同目录下,新建一个login.ts
import axios from './'
namespace Login {
// 用户登录表单
export interface LoginReqForm {
username: string;
password: string;
}
// 登录成功后返回的token
export interface LoginResData {
token: string;
}
}
// 用户登录
export const login = (params: Login.LoginReqForm) => {
// 返回的数据格式可以和服务端约定
return axios.post<Login.LoginResData>('/user/login', params);
}
API接口也定义好了,再来一个页面简单试试:
<script setup>
import { reactive } from 'vue';
import {login} from '@/api/login.js'
const loginForm = reactive({
username: '',
password: ''
})
const Login = async () => {
const data = await login(loginForm)
console.log(data);
}
</script>
<template>
<input v-model="loginForm.username" />
<input v-model="loginForm.password" type="password" />
<button @click="Login">登录</button>
</template>
二、Vue通过Proxy访问不同端口的API
后台API使用Springboot开发,部署端口为8125
1.Vue项目根目录创建vue.config.js
module.exports = {
devServer: {
open: true,
host: "127.0.0.1",
port: 7123,
hot: true,
proxy: {
"/api": {
target: "http://localhost:8125", // 代理地址,这里设置的地址会代替axios中设置的baseURL
changeOrigin: true, // 是否跨域
ws: false, // 如果要代理 websockets,配置这个参数
pathRewrite: {
"^/api": "",
},
},
},
},
};
2.示例请求
apis/index.ts
...
const URL = "/api";
...
apis/register.ts
...
// 用户注册
export const register = (params: RegisterReqForm) => {
return axios.post<number>("/user/register", params);
};
...
pages/login/Login.ts
...
async handleRegisterSubmit() {
const data = await register(this.registerForm);
message.success("用户注册成功");
},
...
根据如上代码可得请求地址为:/api/user/register
,并没有host
代理后的地址为:http://127.0.0.1:8125/user/register
这样就实现前端127.0.0.1:7123
代理请求访问127.0.0.1:8125
引用
1.开箱即用的axios封装:Vue3+ts:https://juejin.cn/post/7107047280133275678#heading-2
2.Proxy error: Could not proxy request... 问题解决:https://blog.csdn.net/ymiandi/article/details/125674056
噢哟,写的不错哦
推荐vue-request,vueuse, ahooks-vue等一些请求库,帮你封装好了各种业务场景。
参考react里为什么不手动请求:
https://blog.skk.moe/post/why-you-should-not-fetch-data-directly-in-use-effect/