调用博主最近登录时间
生活中的HYGGE
Vue3+Ts封装Axios及Vue的Proxy配置

Vue3+Ts封装Axios及Vue的Proxy配置

hygge
2022-12-02 / 2 评论 / 388 阅读 / 正在检测是否收录...

一、开箱即用的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

lb6kjyut.png

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

1

评论 (2)

取消
  1. 头像
    孙泽辉
      Mac OS X 10.15.7 Mac OS X 10.15.7  /  Google Chrome 108.0.0.0 Google Chrome 108.0.0.0

    噢哟,写的不错哦表情

    回复
  2. 头像
    孙泽辉
      Mac OS X 10.15.7 Mac OS X 10.15.7  /  Google Chrome 108.0.0.0 Google Chrome 108.0.0.0

    推荐vue-request,vueuse, ahooks-vue等一些请求库,帮你封装好了各种业务场景。
    参考react里为什么不手动请求:
    https://blog.skk.moe/post/why-you-should-not-fetch-data-directly-in-use-effect/

    回复