摘要:在前端監(jiān)控系統(tǒng)中,或者其他場景下,如果我們需要監(jiān)控當(dāng)前頁面下所有請求狀態(tài)。在通常的監(jiān)控中,監(jiān)控部分的代碼和業(yè)務(wù)部分的代碼是分離的。本文通過的方法實現(xiàn)了一個包可以按需求監(jiān)聽請求。
??在前端監(jiān)控系統(tǒng)中,或者其他場景下,如果我們需要監(jiān)控當(dāng)前頁面下所有請求狀態(tài)??赡芡ǔU埱笙?,我們會選擇在請求的回調(diào)中去處理。這種做法的缺點就是會侵入具體的業(yè)務(wù)代碼。在通常的監(jiān)控中,監(jiān)控部分的代碼和業(yè)務(wù)部分的代碼是分離的。此外,如果存在很多的請求需要被監(jiān)聽,通過侵入具體業(yè)務(wù)代碼,為了減少代碼的重復(fù),也需要封裝監(jiān)聽請求的邏輯。
??本文通過monkey patches的方法實現(xiàn)了一個request-interceptor包,可以按需求監(jiān)聽請求。
??該npm包的項目地址為:https://github.com/forthealll... 歡迎使用。
獲取API請求的狀態(tài)和結(jié)果
monkey patches實現(xiàn)監(jiān)控XMLHttpRequest請求
monkey patches實現(xiàn)監(jiān)控fetch請求
本文的原文在我的博客中:https://github.com/forthealll...
歡迎star
一、獲取API請求和結(jié)果??獲取請求的方式包含了fetch和XMLHttpRequest。比如下面是一個XMLHttpRequest請求的例子:
var client = new XMLHttpRequest(); client.open("POST","http://10.12.72.16:8080/extraInfo" ); client.setRequestHeader("Content-Type", "application/json; charset=utf-8"); client.send(JSON.stringify({}));
??通常我們會通過client上出發(fā)的readystatechange來判斷請求的狀態(tài)以及得到請求的響應(yīng)結(jié)果:
client.onreadystatechange = function () { if (client .readyState==4 &&client.status==200) { console.log(client.responseText);// } }
??XMLHttpRequest的prototype除了onreadystatechange事件外還有其他很多事件,比如onabout、onerror、onload、onloadstart等等事件。如果我們要完整的監(jiān)聽一個請求,那么需要實現(xiàn)完整的實現(xiàn)這些事件:
client.onabout = function(){} client.onerror = function(){} clinet.onload = function(){} client.onloadstart = function(){} ....
??此外如果當(dāng)某一個事件發(fā)生時,需要按順序的實行一系列的函數(shù),這樣會使得事件函數(shù)內(nèi)部越來越復(fù)雜,使得整體項目變的無法維護(hù)。
fetch請求也是同理,因此我們需要合理的封裝監(jiān)聽請求的邏輯。
二、monkey patches實現(xiàn)監(jiān)控XMLHttpRequest請求本文不會具體介紹如何通過monkey patches來封裝監(jiān)聽請求的邏輯,該邏輯已經(jīng)在我的npm包中實現(xiàn),具體可以參考我的開源項目:
https://github.com/forthealll...
本文只介紹如何使用,如有興趣,可以讀一讀具體如何實現(xiàn)這個monkey patches,在目錄的source文件夾中,如有疑問,可以提issue。
該npm包的包名為:req-interceptor。首先來看對于XMLHttpRequest請求如何使用:
import { ajaxIntercept } from "req-interceptor"; //監(jiān)聽 const unregister = ajaxIntercept.register({ requestAbout: function (xhr) { // xhr is real instance of a request console.log(xhr) }, requestError: function (xhr) { // xhr is real instance of a request console.log(xhr) }, requestLoad: function (xhr) { // xhr is real instance of a request console.log(xhr) }, }); //發(fā)送請求 var client = new XMLHttpRequest(); client.open("POST","http://10.12.72.16:8080/extraInfo" ); client.setRequestHeader("Content-Type", "application/json; charset=utf-8"); client.send(JSON.stringify({}));
只需要在發(fā)送請求前先調(diào)用ajaxIntercept.register函數(shù)傳入監(jiān)聽的對象,該函數(shù)會返回一個取消監(jiān)聽的方法。
這樣就監(jiān)聽之后的任意請求,在ajaxIntercept.register中的實際參數(shù)的對象中,對象的屬性是一個函數(shù),參數(shù)為xhr,xhr就是一個被監(jiān)聽的XMLHttpRquest,因此我們可以從xhr中拿到請求的具體響應(yīng)。xhr的一個例子為:
xhr = { readyState: 4 response: "{"success":0}" responseText: "{"success":0}" responseType: "" responseURL: "http://10.12.72.16:8080/extraInfo" responseXML: null status: 201 statusText: "Created" timeout: 0 }
如果我們在取消對于某一個請求的監(jiān)聽,則調(diào)用該返回的
unregister函數(shù),此后請求不會再被監(jiān)聽。
unregister();
此外我們也可以在某一個請求前添加多個監(jiān)聽函數(shù):
import { ajaxIntercept } from "req-interceptor"; //監(jiān)聽 const unregister1 = ajaxIntercept.register({...}); const unregister2 = ajaxIntercept.register({...}); const unregister3 = ajaxIntercept.register({...}); //請求 client.open(url,....)
如果我們想要一次性移除所有的對于請求的監(jiān)聽函數(shù),可以直接調(diào)用:
ajaxIntercept.clear();三、monkey patches實現(xiàn)監(jiān)控fetch請求
對于fetch請求也是一樣的。
import { fetchIntercept } from "req-interceptor"; import { fetchIntercept } from "req-interceptor"; const unregister = fetchIntercept.register({ request: function (url, config) { // Modify the url or config here return [url, config]; }, requestError: function (error) { // Called when an error occured during another "request" interceptor call return Promise.reject(error); }, response: function (response) { // Modify the reponse object return response; }, responseError: function (error) { // Handle an fetch error return Promise.reject(error); } }); // Call fetch to see your interceptors in action. fetch("http://google.com");
不同的是,fetch不像XMLHttpRequest請求那樣,可以監(jiān)聽完整的過程,fetch只有request、requestError、response和responseError這4個屬性可以監(jiān)聽,分別映射請求的參數(shù),請求失敗,請求返回成功,請求返回失敗。
同樣的也可以通過返回函數(shù)來取消監(jiān)聽,以及通過clear函數(shù)來取消所有監(jiān)聽函數(shù)。
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/117280.html
摘要:在前端監(jiān)控系統(tǒng)中,或者其他場景下,如果我們需要監(jiān)控當(dāng)前頁面下所有請求狀態(tài)。在通常的監(jiān)控中,監(jiān)控部分的代碼和業(yè)務(wù)部分的代碼是分離的。本文通過的方法實現(xiàn)了一個包可以按需求監(jiān)聽請求。 ??在前端監(jiān)控系統(tǒng)中,或者其他場景下,如果我們需要監(jiān)控當(dāng)前頁面下所有請求狀態(tài)??赡芡ǔU埱笙?,我們會選擇在請求的回調(diào)中去處理。這種做法的缺點就是會侵入具體的業(yè)務(wù)代碼。在通常的監(jiān)控中,監(jiān)控部分的代碼和業(yè)務(wù)部分的代...
摘要:在前端監(jiān)控系統(tǒng)中,或者其他場景下,如果我們需要監(jiān)控當(dāng)前頁面下所有請求狀態(tài)。在通常的監(jiān)控中,監(jiān)控部分的代碼和業(yè)務(wù)部分的代碼是分離的。本文通過的方法實現(xiàn)了一個包可以按需求監(jiān)聽請求。 ??在前端監(jiān)控系統(tǒng)中,或者其他場景下,如果我們需要監(jiān)控當(dāng)前頁面下所有請求狀態(tài)??赡芡ǔU埱笙?,我們會選擇在請求的回調(diào)中去處理。這種做法的缺點就是會侵入具體的業(yè)務(wù)代碼。在通常的監(jiān)控中,監(jiān)控部分的代碼和業(yè)務(wù)部分的代...
摘要:本項目中采用了進(jìn)行狀態(tài)管理,的主要作用是允許狀態(tài)在不同分支的組件中進(jìn)行傳遞,從而避免了使用原始方法如導(dǎo)致的不同分支組件之間數(shù)據(jù)無法傳遞子組件無法修改父組件狀態(tài)等問題。 項目功能 最近在做一個舊書交易網(wǎng)站,本屬于B/S體系結(jié)構(gòu)的課程作業(yè),但由于采用了新的框架所以躍躍欲試想都記錄下來。 實現(xiàn)一個舊書交易網(wǎng)站,基本功能如下: 實現(xiàn)用戶注冊、登錄功能,用戶注冊時需要填寫必要的信息并驗證,如...
閱讀 2712·2021-10-12 10:12
閱讀 2343·2021-09-02 15:41
閱讀 2577·2019-08-30 15:55
閱讀 1409·2019-08-30 13:05
閱讀 2443·2019-08-29 11:21
閱讀 3542·2019-08-28 17:53
閱讀 3034·2019-08-26 13:39
閱讀 808·2019-08-26 11:50