TypeScript利用TS封裝Axios實戰(zhàn)
簡介
這是TypeScript實戰(zhàn)的第三篇文章。前面兩篇筆者分別介紹了在Vuex和Pinia中怎么使用TypeScript以及Vuex和Pinia的區(qū)別。今天我們再用TypeScript封裝一遍Axios。希望能進一步鞏固TypeScript的基礎知識。
Axios幾個常用類型
在使用TypeScript封裝Axios之前我們先來看看Axios幾個重要的類型。
AxiosRequestConfig
AxiosRequestConfig是我們使用axios發(fā)送請求傳遞參數(shù)的類型。當然它也是我們請求攔截器里面的參數(shù)類型。
axios(config: AxiosRequestConfig)
可以看到,這個config里面的參數(shù)還是挺多的。我們常用的有url、method、params、data、headers、baseURL、timeout。
export interface AxiosRequestConfig {
url?: string;
method?: Method;
baseURL?: string;
transformRequest?: AxiosTransformer | AxiosTransformer[];
transformResponse?: AxiosTransformer | AxiosTransformer[];
headers?: any;
params?: any;
paramsSerializer?: (params: any) => string;
data?: any;
timeout?: number;
timeoutErrorMessage?: string;
withCredentials?: boolean;
adapter?: AxiosAdapter;
auth?: AxiosBasicCredentials;
responseType?: ResponseType;
xsrfCookieName?: string;
xsrfHeaderName?: string;
onUploadProgress?: (progressEvent: any) => void;
onDownloadProgress?: (progressEvent: any) => void;
maxContentLength?: number;
validateStatus?: ((status: number) => boolean) | null;
maxBodyLength?: number;
maxRedirects?: number;
socketPath?: string | null;
httpAgent?: any;
httpsAgent?: any;
proxy?: AxiosProxyConfig | false;
cancelToken?: CancelToken;
decompress?: boolean;
transitional?: TransitionalOptions
}
AxiosInstance
AxiosInstance是我們使用axios實例對象類型。
我們使用axios.create(config?: AxiosRequestConfig)創(chuàng)建出來的對象都是AxiosInstance類型
export interface AxiosInstance {
(config: AxiosRequestConfig): AxiosPromise;
(url: string, config?: AxiosRequestConfig): AxiosPromise;
defaults: AxiosRequestConfig;
interceptors: {
request: AxiosInterceptorManager<AxiosRequestConfig>;
response: AxiosInterceptorManager<AxiosResponse>;
};
getUri(config?: AxiosRequestConfig): string;
request<T = any, R = AxiosResponse<T>> (config: AxiosRequestConfig): Promise<R>;
get<T = any, R = AxiosResponse<T>>(url: string, config?: AxiosRequestConfig): Promise<R>;
delete<T = any, R = AxiosResponse<T>>(url: string, config?: AxiosRequestConfig): Promise<R>;
head<T = any, R = AxiosResponse<T>>(url: string, config?: AxiosRequestConfig): Promise<R>;
options<T = any, R = AxiosResponse<T>>(url: string, config?: AxiosRequestConfig): Promise<R>;
post<T = any, R = AxiosResponse<T>>(url: string, data?: any, config?: AxiosRequestConfig): Promise<R>;
put<T = any, R = AxiosResponse<T>>(url: string, data?: any, config?: AxiosRequestConfig): Promise<R>;
patch<T = any, R = AxiosResponse<T>>(url: string, data?: any, config?: AxiosRequestConfig): Promise<R>;
}可以發(fā)現(xiàn),我們可以使用axios.create、axios.all、axios.spread方法,但是AxiosInstance 上并沒有create、all、spread等方法,那我們的axios到底是什么類型呢?
AxiosStatic
export interface AxiosStatic extends AxiosInstance {
create(config?: AxiosRequestConfig): AxiosInstance;
Cancel: CancelStatic;
CancelToken: CancelTokenStatic;
isCancel(value: any): boolean;
all<T>(values: (T | Promise<T>)[]): Promise<T[]>;
spread<T, R>(callback: (...args: T[]) => R): (array: T[]) => R;
isAxiosError(payload: any): payload is AxiosError;
}
declare const axios: AxiosStatic;可以發(fā)現(xiàn),axios其實是AxiosStatic類型,并且繼承了AxiosInstance類型。所以是兩者的結(jié)合。相較axios.create(config?: AxiosRequestConfig)創(chuàng)建出來的實例對象,axios功能是更強大的。
AxiosResponse
AxiosResponse是非常重要的,我們的axios請求返回值類型都是AxiosResponse類型。并且我們可以發(fā)現(xiàn)AxiosResponse是一個接口泛型,這個泛型會應用到后端返回的data上。所以這塊我們可以根據(jù)后端接口返回定義不同的類型傳遞進去。后面筆者在封裝常用方法的時候會細說。
export interface AxiosResponse<T = any> {
data: T;
status: number;
statusText: string;
headers: any;
config: AxiosRequestConfig;
request?: any;
}
AxiosError
AxiosError這個類型也是我們必須要知道的。在我們響應攔截器里面的錯誤就是AxiosError類型。
export interface AxiosError<T = any> extends Error {
config: AxiosRequestConfig;
code?: string;
request?: any;
response?: AxiosResponse<T>;
isAxiosError: boolean;
toJSON: () => object;
}
說完了Axios的幾個常用類型,接下來我們正式開始使用TS來封裝我們的Axios。
基礎封裝
首先我們實現(xiàn)一個最基本的版本,實例代碼如下:
// index.ts
import axios from 'axios'
import type { AxiosInstance, AxiosRequestConfig, AxiosResponse } from 'axios'
class Request {
// axios 實例
instance: AxiosInstance
// 基礎配置,url和超時時間
baseConfig: AxiosRequestConfig = {baseURL: "/api", timeout: 60000}
constructor(config: AxiosRequestConfig) {
// 使用axios.create創(chuàng)建axios實例
this.instance = axios.create(Object.assign(this.baseConfig, config))
}
// 定義請求方法
public request(config: AxiosRequestConfig): Promise<AxiosResponse> {
return this.instance.request(config)
}
}
export default Request
在實際項目中有了基本的請求方法還是遠遠不夠的,我們還需要封裝攔截器和一些常用方法。
攔截器封裝
攔截器封裝只需要在類中對axios.create()創(chuàng)建的實例調(diào)用interceptors下的兩個攔截器即可,
實例代碼如下:
// index.ts
constructor(config: AxiosRequestConfig) {
this.instance = axios.create(Object.assign(this.baseConfig, config))
this.instance.interceptors.request.use(
(config: AxiosRequestConfig) => {
// 一般會請求攔截里面加token
const token = localStorage.getItem("token")
config.headers["Authorization"] = token;
return config
},
(err: any) => {
return Promise.reject(err)
},
)
this.instance.interceptors.response.use(
(res: AxiosResponse) => {
// 直接返回res,當然你也可以只返回res.data
return res
},
(err: any) => {
// 這里用來處理http常見錯誤,進行全局提示
let message = "";
switch (err.response.status) {
case 400:
message = "請求錯誤(400)";
break;
case 401:
message = "未授權(quán),請重新登錄(401)";
// 這里可以做清空storage并跳轉(zhuǎn)到登錄頁的操作
break;
case 403:
message = "拒絕訪問(403)";
break;
case 404:
message = "請求出錯(404)";
break;
case 408:
message = "請求超時(408)";
break;
case 500:
message = "服務器錯誤(500)";
break;
case 501:
message = "服務未實現(xiàn)(501)";
break;
case 502:
message = "網(wǎng)絡錯誤(502)";
break;
case 503:
message = "服務不可用(503)";
break;
case 504:
message = "網(wǎng)絡超時(504)";
break;
case 505:
message = "HTTP版本不受支持(505)";
break;
default:
message = `連接出錯(${err.response.status})!`;
}
// 這里錯誤消息可以使用全局彈框展示出來
// 比如element plus 可以使用 ElMessage
ElMessage({
showClose: true,
message: `${message},請檢查網(wǎng)絡或聯(lián)系管理員!`,
type: "error",
});
// 這里是AxiosError類型,所以一般我們只reject我們需要的響應即可
return Promise.reject(err.response)
},
)
}在這里我們分別對請求攔截器和響應攔截器做了處理。在請求攔截器我們給請求頭添加了token。
在響應攔截器,我們返回了整個response對象,當然你也可以只返回后端返回的response.data,這里可以根據(jù)個人喜好來處理。其次對http錯誤進行了全局處理。
常用方法封裝
在基礎封裝的時候我們封裝了一個request通用方法,其實我們還可以更具體的封裝get、post、put、delete方法,讓我們使用更方便。
并且,我們前面分析到,AxiosResponse其實是一個泛型接口,他可以接受一個泛型并應用到我們的data上。所以我們可以在這里再定義一個后端通用返回的數(shù)據(jù)類型。
比如假設我們某個項目后端接口不管請求成功與失敗,返回的結(jié)構(gòu)永遠是code、message、results的話我們可以定義一個這樣的數(shù)據(jù)類型。
type Result<T> = {
code: number,
message: string,
result: T
}然后傳遞個各個方法:
public get<T = any>(
url: string,
config?: AxiosRequestConfig
): Promise<AxiosResponse<Result<T>>> {
return this.instance.get(url, config);
}
public post<T = any>(
url: string,
data?: any,
config?: AxiosRequestConfig
): Promise<AxiosResponse<Result<T>>> {
return this.instance.post(url, data, config);
}
public put<T = any>(
url: string,
data?: any,
config?: AxiosRequestConfig
): Promise<AxiosResponse<Result<T>>> {
return this.instance.put(url, data, config);
}
public delete<T = any>(
url: string,
config?: AxiosRequestConfig
): Promise<AxiosResponse<Result<T>>> {
return this.instance.delete(url, config);
}這樣當我們調(diào)用接口的時候就可以看到我們返回的data的類型啦。就是我們定義的Result類型。

所以我們可以直接得到自動提示:

上面調(diào)用接口的時候并沒有傳遞接口數(shù)據(jù)類型,所以我們的result是any類型,要想要每個接口都有類型提示,我們還需要給方法傳遞泛型。
我們再改進下,我們再定義一個login接口返回值類型loginType
type loginType = {
token: string;
};
然后再調(diào)用方法的地方傳遞進去,然后我們再看看返回值data的類型。

可以看到他是Result<loginType>類型,這個loginType就是result的類型。
所以我們的result還可以進一步的得到提示

當然每個接口都定義返回值類型固然好,但是會大大加大前端的工作量。我們在寫請求方法的時候也可以不傳遞接口返回值類型,這樣result的類型就是any。這個可以根據(jù)自身項目需求來選擇使用。
看到這小伙伴們是不是都弄懂了呢?如還有疑問歡迎留言。
總結(jié)
說了這么多,有些小伙伴們可能有點暈了,下面筆者總結(jié)下整個axios的封裝。
// index.ts
import axios from "axios";
import type { AxiosInstance, AxiosRequestConfig, AxiosResponse } from "axios";
type Result<T> = {
code: number;
message: string;
result: T;
};
class Request {
// axios 實例
instance: AxiosInstance;
// 基礎配置,url和超時時間
baseConfig: AxiosRequestConfig = { baseURL: "/api", timeout: 60000 };
constructor(config: AxiosRequestConfig) {
// 使用axios.create創(chuàng)建axios實例
this.instance = axios.create(Object.assign(this.baseConfig, config));
this.instance.interceptors.request.use(
(config: AxiosRequestConfig) => {
// 一般會請求攔截里面加token
const token = localStorage.getItem("token");
config.headers["Authorization"] = token;
return config;
},
(err: any) => {
return Promise.reject(err);
}
);
this.instance.interceptors.response.use(
(res: AxiosResponse) => {
// 直接返回res,當然你也可以只返回res.data
return res;
},
(err: any) => {
// 這里用來處理http常見錯誤,進行全局提示
let message = "";
switch (err.response.status) {
case 400:
message = "請求錯誤(400)";
break;
case 401:
message = "未授權(quán),請重新登錄(401)";
// 這里可以做清空storage并跳轉(zhuǎn)到登錄頁的操作
break;
case 403:
message = "拒絕訪問(403)";
break;
case 404:
message = "請求出錯(404)";
break;
case 408:
message = "請求超時(408)";
break;
case 500:
message = "服務器錯誤(500)";
break;
case 501:
message = "服務未實現(xiàn)(501)";
break;
case 502:
message = "網(wǎng)絡錯誤(502)";
break;
case 503:
message = "服務不可用(503)";
break;
case 504:
message = "網(wǎng)絡超時(504)";
break;
case 505:
message = "HTTP版本不受支持(505)";
break;
default:
message = `連接出錯(${err.response.status})!`;
}
// 這里錯誤消息可以使用全局彈框展示出來
// 比如element plus 可以使用 ElMessage
ElMessage({
showClose: true,
message: `${message},請檢查網(wǎng)絡或聯(lián)系管理員!`,
type: "error",
});
// 這里是AxiosError類型,所以一般我們只reject我們需要的響應即可
return Promise.reject(err.response);
}
);
}
// 定義請求方法
public request(config: AxiosRequestConfig): Promise<AxiosResponse> {
return this.instance.request(config);
}
public get<T = any>(
url: string,
config?: AxiosRequestConfig
): Promise<AxiosResponse<Result<T>>> {
return this.instance.get(url, config);
}
public post<T = any>(
url: string,
data?: any,
config?: AxiosRequestConfig
): Promise<AxiosResponse<Result<T>>> {
return this.instance.post(url, data, config);
}
public put<T = any>(
url: string,
data?: any,
config?: AxiosRequestConfig
): Promise<AxiosResponse<Result<T>>> {
return this.instance.put(url, data, config);
}
public delete<T = any>(
url: string,
config?: AxiosRequestConfig
): Promise<AxiosResponse<Result<T>>> {
return this.instance.delete(url, config);
}
}
export default Request;到此這篇關(guān)于TypeScript利用TS封裝Axios實戰(zhàn)的文章就介紹到這了,更多相關(guān)TypeScript封裝Axios內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
JavaScript高級程序設計 閱讀筆記(十九) js表格排序
js表格排序?qū)崿F(xiàn)代碼,需要的朋友可以參考下2012-08-08

