掃二維碼與項(xiàng)目經(jīng)理溝通
我們?cè)谖⑿派?4小時(shí)期待你的聲音
解答本文疑問/技術(shù)咨詢/運(yùn)營(yíng)咨詢/技術(shù)建議/互聯(lián)網(wǎng)交流
作者:京東物流 李菲菲

現(xiàn)有的大部分監(jiān)控方案都是針對(duì)服務(wù)端的,而針對(duì)前端的監(jiān)控很少,諸如線上頁面的白屏?xí)r間是多少、靜態(tài)資源的加載情況如何、接口請(qǐng)求耗時(shí)好久、什么時(shí)候掛掉了、為什么掛掉,這些都不清楚。
同時(shí),在產(chǎn)品推廣過程中,經(jīng)常需要統(tǒng)計(jì)頁面的使用情況及用戶行為,從而可以從運(yùn)營(yíng)和產(chǎn)品的角度去了解用戶群體,進(jìn)而迭代升級(jí)產(chǎn)品,使其更加貼近用戶,為業(yè)務(wù)的擴(kuò)展提供更多可能性。
因而,我們需要一個(gè)前端的頁面監(jiān)控系統(tǒng),持續(xù)監(jiān)控和預(yù)警頁面性能的狀況,并且在發(fā)現(xiàn)瓶頸時(shí)用于指導(dǎo)優(yōu)化工作。
前端監(jiān)控主要包含兩大塊:性能監(jiān)控及異常監(jiān)控
簡(jiǎn)單看一下,從輸入url到頁面加載完成的過程如下:
首先需要通過 DNS(域名解析系統(tǒng))將 URL 解析為對(duì)應(yīng)的 IP 地址,然后與這個(gè) IP 地址確定的那臺(tái)服務(wù)器建立起 TCP 網(wǎng)絡(luò)連接,隨后我們向服務(wù)端拋出 HTTP 請(qǐng)求,服務(wù)端處理完我們的請(qǐng)求之后,把目標(biāo)數(shù)據(jù)放在 HTTP 響應(yīng)里返回給客戶端,拿到響應(yīng)數(shù)據(jù)的瀏覽器就可以開始走一個(gè)渲染的流程。渲染完畢,頁面便呈現(xiàn)給了用戶。
我們可以將這個(gè)過程分為如下的過程片段:
從輸入url到用戶可以使用頁面的全過程時(shí)間統(tǒng)計(jì),會(huì)返回一個(gè)PerformanceTiming對(duì)象,單位均為毫秒。
關(guān)于performace,已經(jīng)在《從前端角度淺談性能》中進(jìn)行過介紹,,下面再?gòu)?qiáng)調(diào)一下:
各階段的性能耗時(shí)可以通過API:window.performance來獲取,對(duì)應(yīng)的具體方法有:performance.timing、performance.getEntriesByType(‘resource’)、performance.navigation等。
如上,開發(fā)者可以通過performance中各階段的時(shí)間戳,分別獲取到 頁面各階段的性能指標(biāo),具體的個(gè)靜態(tài)資源的加載耗時(shí)、及 頁面是否重定向和重定向耗時(shí)。
按觸發(fā)順序排列所有屬性:
const { timing, navigation } = window.performance
const loadPageInfo = {};
// 頁面加載類型,區(qū)分第一次load還是reload, 0初次加載、1重加載
loadPageInfo.loadType = navigation.type;
// 頁面加載完成的時(shí)間 - 幾乎代表了用戶等待頁面白屏的時(shí)間
loadPageInfo.loadPage = timing.loadEventEnd - timing.navigationStart;
// 重定向的時(shí)間
loadPageInfo.redirect = timing.redirectEnd - timing.redirectStart;
// 卸載頁面的時(shí)間
loadPageInfo.unloadEvent = timing.unloadEventEnd - timing.unloadEventStart;
// 查詢 DNS 本地緩存的時(shí)間
loadPageInfo.appCache = timing.domainLookupStart - timing.fetchStart;
// 【重要】DNS 查詢時(shí)間
// 頁面內(nèi)是不是使用了太多不同的域名,導(dǎo)致域名查詢的時(shí)間太長(zhǎng)?推薦 DNS 預(yù)加載。
// 可使用 HTML5 Prefetch 預(yù)查詢 DNS
loadPageInfo.lookupDomain = timing.domainLookupEnd - timing.domainLookupStart;
// HTTP(TCP)建立連接完成握手的時(shí)間
loadPageInfo.connect = timing.connectEnd - timing.connectStart;
// 【重要】HTTP請(qǐng)求及獲取 文檔內(nèi)容的時(shí)間
loadPageInfo.request = timing.responseEnd - timing.responseStart;
// 【重要】前一個(gè)頁面 unload 到 HTTP獲取到 頁面第一個(gè)字節(jié)的時(shí)間
// 【原因】這可以理解為用戶拿到你的資源占用的時(shí)間,推薦 加異地機(jī)房,加 CDN 處理,加寬帶,加 CPU 運(yùn)算速度
// TTFB 即 Time To First Byte
loadPageInfo.ttfb = timing.responseStart - timing.navigationStart;
// 解析 DOM 樹結(jié)構(gòu)的時(shí)間
loadPageInfo.domReady = timing.domComplete - timing.responseEnd;
// 【重要】執(zhí)行 onload 回調(diào)函數(shù)的時(shí)間
// 【原因】是否太多不必要的操作都放在 onload 回調(diào)函數(shù)里執(zhí)行了,推薦 延遲加載、按需加載的策略
loadPageInfo.loadEvent = timing.loadEventEnd - timing.loadE前端需要監(jiān)控的錯(cuò)誤主要有兩類:
// 重寫console.error,可以捕獲更全面的報(bào)錯(cuò)信息
var oldError = console.error;
console.error = function(tempErrorMsg){
var errorMsg = ( arguments[0] && arguments[0].message ) || tempErrorMsg;
var lineNumber = 0;
var columnNumber = 0;
var errorStack = arguments[0] && arguments[0].stack;
if( !errorStack ){
saveJSError( 'console_error', errorMsg, '', lineNumber, columnNumber, 'CustomizeError: ' + errorMsg );
}else{
saveJSError( 'console_error', errorMsg, '', lineNumber, columnNumber, errorStack );
}
return oldError.apply( console, arguments );
通過對(duì)error事件的監(jiān)聽,可以捕捉到 js語法 及 資源加載 的錯(cuò)誤。根據(jù) event.target.src / href 來判斷是否為資源加載錯(cuò)誤。
window.addEventListener( 'error', function(e){
var errorMsg = e.error && e.error.message,
errorStack = e.error && e.error.stack,
pageUrl = e.filename,
lineNumber = e.lineno,
columnNumber = e.colno;
saveJSError( 'on_error', errorMsg, pageUrl, lineNumber, columnNumber, errorStack );
} );// 捕獲未處理的Promise錯(cuò)誤
window.onunhandledrejection = function(e){
var errorMsg = '';
var errorStack = '';
if( typeof e.reason === 'object' ){
errorMsg = e.reason.message;
errorStack = e.reason.stack;
}else{
errorMsg = e.reason;
errorStack = '';
}
saveJSError( 'on_error', errorMsg, '', 0, 0, 'UncaughtInPromiseError: ' + errorStack );
}
以上,通過簡(jiǎn)單的js代碼,即可實(shí)現(xiàn)對(duì)頁面性能與異常的監(jiān)控與數(shù)據(jù)上報(bào),后續(xù)還需要相應(yīng)具體的平臺(tái)匯總,及相應(yīng)的業(yè)務(wù)所需數(shù)據(jù)(如PV、UV等)的計(jì)算,才能真正實(shí)現(xiàn)對(duì)產(chǎn)品的頁面數(shù)據(jù)呈現(xiàn),用于業(yè)務(wù)擴(kuò)展及宣導(dǎo)。
上述代碼,實(shí)現(xiàn)了對(duì)頁面性能及異常的監(jiān)控,但其實(shí)前端的監(jiān)控還包括了請(qǐng)求接口的監(jiān)控與埋點(diǎn)的實(shí)現(xiàn),后續(xù)將陸續(xù)推出,敬請(qǐng)期待。

我們?cè)谖⑿派?4小時(shí)期待你的聲音
解答本文疑問/技術(shù)咨詢/運(yùn)營(yíng)咨詢/技術(shù)建議/互聯(lián)網(wǎng)交流