vue項目中使用fetch的實現(xiàn)方法
fetch的由來和定義
fetch的由來
眾所周知,傳統(tǒng) Ajax (指 XMLHttpRequest)是最早出現(xiàn)的發(fā)送異步請求技術(shù),其核心是使用XMLHttpRequest對象。但是它也存在一些令人頭疼的問題:XHR 是一個設(shè)計粗糙的 API,不符合關(guān)注分離的原則;配置和調(diào)用方式非常混亂,而且基于事件的異步模型寫起來也沒有現(xiàn)代的 Promise,generator/yield,async/await 友好。而Fetch 的出現(xiàn)就是為了解決 XHR 存在的問題。
fetch的定義和使用
MDN中的描述:
Fetch API 提供了一個獲取資源的接口(包括跨域請求)。任何使用過 XMLHttpRequest 的人都能輕松上手,但新的API提供了更強大和靈活的功能集。but 因為凄慘的兼容性,讓這個東東用起來比較困難。那我可以自己封裝一下,對于不支持fetch的瀏覽器便使用ajax 代替(見下文)。
Fetch 的核心在于對 HTTP 接口的抽象,包括 Request,Response,Headers,Body,以及用于初始化異步請求的 global fetch。其中,global fetch方法的語法定義:
fetch(input[, init]);
input:定義要獲取的資源??梢允且粋€資源的 URL 字符串,也可以是一個 Request 對象。
init:可選,一個配置項對象,包括所有對請求的設(shè)置。包括:method,headers,body,mode,credentials等返回值:Promise
切記一點:Fetch是基于promise設(shè)計的,它不是ajax的進一步封裝,而是原生js API,沒有使用XMLHttpRequest對象。
fetch的優(yōu)點和缺點
優(yōu)點:
1. 語法簡潔,更加語義化
2. 基于標準 Promise 實現(xiàn),支持 async/await
3. 同構(gòu)方便,更加底層,提供的API豐富(request, response, body , headers)5. 脫離了XHR,是ES規(guī)范里新的實現(xiàn)方式
缺點:
1. fetch只對網(wǎng)絡(luò)請求報錯,對400,500都當(dāng)做成功的請求,服務(wù)器返回 400,500 錯誤碼時并不會 reject。
2. fetch默認不會帶cookie,需要添加配置項: credentials: 'include'。
3. fetch不支持abort,不支持超時控制,造成了流量的浪費。
4. fetch沒有辦法原生監(jiān)測請求的進度,而XHR可以
補充知識點:
Fetch的mode配置項有3個取值:
same-origin:該模式是不允許跨域的,它需要遵守同源策略;
cors: 該模式支持跨域請求,顧名思義它是以CORS的形式跨域;
no-cors: 該模式用于跨域請求但是服務(wù)器不帶CORS響應(yīng)頭,也就是服務(wù)端不支持CORS;目前,針對跨域請求,cors模式是常見的實現(xiàn)。
vue項目中完美封裝fetch
話不多少,直接附上代碼。
env.js文件,如下:
/**
* baseUrl: 域名地址
* routerMode: 路由模式
*/
let baseUrl = '';
let routerMode = 'history';
if (process.env.NODE_ENV == 'development') {
baseUrl = 'http://localhost:3000';
}else{
baseUrl = 'http://xxxx這里是線上地址xxx';
}
export { baseUrl, routerMode }
fetch.js文件,如下:
import { baseUrl } from './env'
export default async(url = '', data = {}, type = 'GET', method = 'fetch') => {
type = type.toUpperCase();
url = baseUrl + url;
// 此處規(guī)定get請求的參數(shù)使用時放在data中,如同post請求
if (type == 'GET') {
let dataStr = '';
Object.keys(data).forEach(key => {
dataStr += key + '=' + data[key] + '&';
})
if (dataStr !== '') {
dataStr = dataStr.substr(0, dataStr.lastIndexOf('&'));
url = url + '?' + dataStr;
}
}
// 對于支持fetch方法的瀏覽器,處理如下:
if (window.fetch && method == 'fetch') {
let requestConfig = {
// fetch默認不會帶cookie,需要添加配置項credentials允許攜帶cookie
credentials: 'include',
method: type,
headers: {
'Accept': 'application/json',
'Content-Type': 'application/json'
},
mode: "cors", // 以CORS的形式跨域
cache: "force-cache"
}
if (type == 'POST') {
Object.defineProperty(requestConfig, 'body', {
value: JSON.stringify(data)
})
}
try {
const response = await fetch(url, requestConfig);
const responseJson = await response.json();
return responseJson
} catch (error) {
throw new Error(error)
}
} else { // 對于不支持fetch的瀏覽器,便自動使用 ajax + promise
return new Promise((resolve, reject) => {
let requestObj;
if (window.XMLHttpRequest) {
requestObj = new XMLHttpRequest();
} else {
requestObj = new ActiveXObject; // 兼容IE
}
let sendData = '';
if (type == 'POST') {
sendData = JSON.stringify(data);
}
requestObj.open(type, url, true);
requestObj.setRequestHeader("Content-type", "application/x-www-form-urlencoded");
requestObj.send(sendData);
requestObj.onreadystatechange = () => {
if (requestObj.readyState == 4) {
if (requestObj.status == 200) {
let obj = requestObj.response
if (typeof obj !== 'object') {
obj = JSON.parse(obj);
}
resolve(obj)
} else {
reject(requestObj)
}
}
}
})
}
}
以上代碼,親測有效。Over, thanks !希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
詳解vue beforeRouteEnter 異步獲取數(shù)據(jù)給實例問題
這篇文章主要介紹了vue beforeRouteEnter 異步獲取數(shù)據(jù)給實例問題,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2019-08-08
解決element-ui的table表格控件表頭與內(nèi)容列不對齊問題
這篇文章主要介紹了解決element-ui的table表格控件表頭與內(nèi)容列不對齊問題,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-07-07
vue 解決移動端彈出鍵盤導(dǎo)致頁面fixed布局錯亂的問題
今天小編就為大家分享一篇vue 解決移動端彈出鍵盤導(dǎo)致頁面fixed布局錯亂的問題,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2019-11-11
關(guān)于Vue中echarts響應(yīng)式頁面變化resize()的用法介紹
Vue項目中開發(fā)數(shù)據(jù)大屏,使用echarts圖表根據(jù)不同尺寸的屏幕進行適配,resize()可以調(diào)用echarts中內(nèi)置的resize函數(shù)進行自適應(yīng)縮放,本文將給大家詳細介紹resize()的用法,需要的朋友可以參考下2023-06-06

