成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

關(guān)于首屏時間采集自動化的解決方案

1treeS / 862人閱讀

摘要:關(guān)于首屏首屏時間是指從轉(zhuǎn)向該頁面到屏幕中該頁面所有內(nèi)容都可見時的時間。如在事件處理函數(shù)中,計算高度,如果大于屏幕高度則意味著首屏的結(jié)構(gòu)已渲染完畢,開始計算首屏時間。

關(guān)于首屏

首屏時間是指從轉(zhuǎn)向該頁面到屏幕中該頁面所有內(nèi)容都可見時的時間。已經(jīng)有太多的關(guān)于首屏時間的計算,在本文中并不重復闡述這些已經(jīng)被提出或者實現(xiàn)的方案,而旨在探索與討論更多的首屏自動化采集方案,擴大思考范圍,你我思想之間互相碰撞往往可以激起更多的稀奇古怪的解決方案,這也正是我寫這篇文章的目的。

通過瀏覽器調(diào)試工具,我們可以清晰的看出頁面資源加載時序圖:

先是html頁面加載,token進行詞法、語法解析后開始加載靜態(tài)資源并執(zhí)行相關(guān)腳本,開始構(gòu)建DOM樹、render樹和CSSOM數(shù),最后加載圖片,用戶看到完整的網(wǎng)頁。

雖然瀏覽器有著各自的優(yōu)化的解決方案,但是大多數(shù)情況下圖片往往是最后加載完畢,這不僅僅是由于圖片的大小相對較大,而且圖片的加載與否與DOM結(jié)構(gòu)有著很大的關(guān)系。DOM是否構(gòu)建完畢,render樹中是否渲染以及其他的圖片加載策略有關(guān)系可能都會影響圖片加載時序。因此在首屏時間的計算中,我們是以最終首屏圖片的加載時間為節(jié)點計算的。

首屏計算 原則1 首屏計算模塊不應該耦合業(yè)務線

一般而言,首屏計算作為一個抽離出的js腳本多帶帶引用,這個模塊盡量不暴露API給開發(fā)者使用,所有的采集端任務都由該模塊完成。這句話可能聽起來像一句廢話,但還是有很多情況可能需要業(yè)務人員來進行首屏渲染時間的判斷的,下面將針對這個情形舉一個實際的場景:

隨著MVVM模式的興起,前端異步渲染逐漸流行起來,前端編碼逐漸由面向jQuery編程轉(zhuǎn)向為面向Vue編程??墒鞘褂肰ue編寫的業(yè)務代碼在本地打包后僅僅是一個bundle,此時的HTML文件中只是一個

的占位符而已,那么首屏時間計算模塊該如何準確的計算首屏時間呢?因此首屏時間計算模塊必須知道首屏的DOM結(jié)構(gòu)渲染完畢的時間節(jié)點,在這個節(jié)點時刻進行計算首屏范圍內(nèi)的圖片加載時間。 可是如何獲取首屏DOM結(jié)構(gòu)渲染完畢的時間節(jié)點呢?這就需要業(yè)務開發(fā)人員制定。在更新vue實例的data屬性后,通知首屏計算模塊此時DOM接口已渲染完畢,開始計算首屏時間。

MVVM開發(fā)模式下,首屏時間的計算已經(jīng)耦合了業(yè)務代碼,雖然可以在保證首屏時間的準確性,但卻給開發(fā)者帶來了一些可觀判斷邏輯,而這些判斷往往會困擾新入職的同志們,因此我們的目標之一就是解決需要手動打點進行首屏時間計算的現(xiàn)狀。

原則2 性能與準確性的權(quán)衡

業(yè)界有個通過canvas截屏并通過輪詢對比不同時間點截屏圖片之間某幾個隨機像素點,從而判斷首屏是否加載完畢。這種方式雖然科學,但是估計沒有幾個公司會采用這種方案。通過canvas截屏這個操作對硬件的要求可能就比較高,而且需要進行額外的像素運算,因此性能肯定很差。其實這種場景在工程領(lǐng)域經(jīng)常出現(xiàn),工程不同于科學那般嚴謹,我們只需要找到給定條件的最優(yōu)解即可,做工程也就是在做trade off。因此這種對比方案我們也必須摒棄。

實現(xiàn)

再次強調(diào),由開發(fā)者打點首屏DOM渲染完畢進行首屏時間計算的方式是相對準確的方式,因此我們后續(xù)討論的自動化計算首屏時間的準確性都是基于此標準進行對比說明,因為自動化計算肯定是沒有人工干預準確的,這一點毫無疑問。

輪訓采集大法

仍然是輪訓,不同的是在每次輪詢中執(zhí)行一些操作:

獲取首屏的所有圖片(包括IMG標簽與css相關(guān)屬性)

綁定首屏圖片的onload和onerror事件,每次輪詢不會重復綁定已綁定的圖片

相同圖片不需重復綁定事件偵聽,否則會與 2 中的每次輪詢混淆

圖片的事件處理函數(shù)執(zhí)行打點信息并統(tǒng)計圖片加載狀態(tài),同時比對時間戳得到最遲加載的時間

具體的實現(xiàn)中,需要特別注意首屏出現(xiàn)的相同圖片的情況。筆者起初在獲取首屏圖片中簡單計算圖片的url數(shù)組,存儲重復圖片的個數(shù),并且與該圖片的加載狀態(tài)綁定在一起。如首屏中出現(xiàn)了3張相同的圖片,那么在該圖片onload或onerror中對已加載圖片的數(shù)量做 加3 處理,否則導致最終的 已加載圖片總數(shù) 與 首屏圖片總數(shù) 不相等的情況發(fā)生。這種實現(xiàn)導致邏輯非常的差,且實現(xiàn)復雜。后通過存儲圖片所在的DOM對象數(shù)組實現(xiàn)更為簡單的圖片狀態(tài)判斷,更加已讀。

偽代碼如下:

// totalCounter為輪詢的總時間
// DemandCounter是規(guī)定的輪詢總時間,為3000ms
// imgsLoadedCount則為首屏已加載的圖片數(shù)量
// lastImageLoadedStamp為最后加載的圖片時間戳
function checkFirstScreenDomReady(){
    if(totalCounter >= DemandCounter){
        // ...
        var stamps = Object.keys(pools),
        len = stamps.length,
        i = 0,
        it; 
        finalImgCount = pools[stamps[len - 1]].imgLen; 
        pollEnd = true;

        for(;i= finalImgCount){
                self.onRecord = true;
                _perfQueue._firstScreenLoadEnd = lastImageLoadedStamp;
                firstScreen.firstScreenLoadEnd = lastImageLoadedStamp;
                firstScreen.duaring = lastImageLoadedStamp - performance.timing.navigationStart;
                
                reportData(firstScreen);
                return;
            }
        }
        return;
    }
    
    var imgEls = getImage();

    imgEls.forEach(function(el) {
        if(!imgLoadedHash.get(el)){
            var img = new Image();
            imgLoadedHash.put(el,{
                loaded: true,
            });
            img.onload = OnLoad;
            img.onerror = OnError;
            img.src = el.__src;
        }
    });
    pools[totalCounter+""] = {
        imgLen: imgEls.length,
        stamp: Date.now(),
        imgsLoadedCount: imgsLoadedCount
    };
    totalCounter += timeout;
}
watch dog采集

利用Mutation Observer API進行偵聽 內(nèi)容框的DOM事件,判斷首屏DOM結(jié)構(gòu)是否完備;如果構(gòu)建完畢則偵聽首屏范圍內(nèi)的圖片加載事件,計算首屏時間。

watch dog需要知曉合適首屏DOM構(gòu)建完畢。這需要首屏計算模塊主動插入一個打點標簽

,將業(yè)務代碼放置在標簽內(nèi)部(這個步驟最好放在發(fā)布階段,由腳手架操作)。通過mutation 偵聽 .j_collector_container 容器的DOM子孫節(jié)點變化。如在observe事件處理函數(shù)中,計算 .j_collector_container 高度,如果大于屏幕高度則意味著首屏的DOM結(jié)構(gòu)已渲染完畢,開始計算首屏時間。

在計算 .j_collector_container 高度時,最好采用限流策略,防止短時間內(nèi)計算多次容器的布局信息,這也是無可奈何之舉。

此處的偽代碼如下:

// 記錄首屏DOM元素的位置信息
var firstScreenDomReady = false;
var callback = function(records){
    if(firstScreenDomReady)
        return;

    // 此處需做throttle 處理
    for(var i=0,len=records.length;i= screenHeight){
            firstScreenDomReady = true;
            recordFirstScreenLoad();
            return;
        }
    }
};
    
var mo = new MutationObserver(callback);

var option = {
    "childList": true,
    "subtree": true
};

var collectWrapper = document.querySelector(".j_collector_wrapper");
if(collectWrapper.getBoundingClientRect().height < win.innerHeight){
    mo.observe(collectWrapper, option);
}else{
    setTimeout(function(){
        recordFirstScreenLoad();
    });
}
總結(jié)

不管采用哪種方式,計算出來的首屏時間都不是準確的。而且在每種實現(xiàn)中都需要通過JS引擎與渲染引擎的bridge進行通信執(zhí)行耗時的操作,如getBoundingClientRect和訪問offsetTop屬性導致relayout。不過這也是沒有辦法的辦法,在瀏覽器不提供相關(guān)首屏API的前提下我們只有這么做。

另外,對比這三種實現(xiàn)(開發(fā)者手動打點、輪訓、watch dog采集),針對一個復雜的電商首屏做了性能測試,該頁面首屏部分有7個非常復雜的子組件,得到如下結(jié)果:

結(jié)果也符合我們的預期。

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/92843.html

相關(guān)文章

  • 關(guān)于首屏時間采集動化解決方案

    摘要:關(guān)于首屏首屏時間是指從轉(zhuǎn)向該頁面到屏幕中該頁面所有內(nèi)容都可見時的時間。如在事件處理函數(shù)中,計算高度,如果大于屏幕高度則意味著首屏的結(jié)構(gòu)已渲染完畢,開始計算首屏時間。 關(guān)于首屏 首屏時間是指從轉(zhuǎn)向該頁面到屏幕中該頁面所有內(nèi)容都可見時的時間。已經(jīng)有太多的關(guān)于首屏時間的計算,在本文中并不重復闡述這些已經(jīng)被提出或者實現(xiàn)的方案,而旨在探索與討論更多的首屏自動化采集方案,擴大思考范圍,你我思想之間...

    Rindia 評論0 收藏0
  • [聊一聊系列]聊一聊前端速度統(tǒng)計(性能統(tǒng)計)那些事兒

    摘要:性能統(tǒng)計有助于幫我們檢測網(wǎng)站的用戶體驗。這樣,我們就輕輕松松的統(tǒng)計到了首屏時間。下一章,我們將繼續(xù)聊聊百度移動版首頁那些事。 歡迎大家收看聊一聊系列,這一套系列文章,可以幫助前端工程師們了解前端的方方面面(不僅僅是代碼): https://segmentfault.com/blog/frontenddriver 上一篇文章我們討論了,如何進行前端日志打點統(tǒng)計: https://segm...

    gclove 評論0 收藏0
  • [聊一聊系列]聊一聊前端速度統(tǒng)計(性能統(tǒng)計)那些事兒

    摘要:性能統(tǒng)計有助于幫我們檢測網(wǎng)站的用戶體驗。這樣,我們就輕輕松松的統(tǒng)計到了首屏時間。下一章,我們將繼續(xù)聊聊百度移動版首頁那些事。 歡迎大家收看聊一聊系列,這一套系列文章,可以幫助前端工程師們了解前端的方方面面(不僅僅是代碼): https://segmentfault.com/blog/frontenddriver 上一篇文章我們討論了,如何進行前端日志打點統(tǒng)計: https://segm...

    Karuru 評論0 收藏0
  • 讓前端監(jiān)控數(shù)據(jù)采集更高效

    摘要:如何在新的技術(shù)背景下讓前端數(shù)據(jù)采集工作更加完善高效,是本文討論的重點。具體來說,我們對前端的數(shù)據(jù)采集具體主要分為路由切換性能資源錯誤日志上報路由切換等前端技術(shù)的快速發(fā)展使單頁面應用盛行。 隨著業(yè)務的快速發(fā)展,我們對生產(chǎn)環(huán)境下的問題感知能力越來越關(guān)注。作為距離用戶最近的一層,前端的表現(xiàn)是否可靠、穩(wěn)定、好用,很大程度上決定著用戶對整個產(chǎn)品的體驗和感受。因此,對于前端的監(jiān)控不容忽視。 搭建一...

    Half 評論0 收藏0

發(fā)表評論

0條評論

最新活動
閱讀需要支付1元查看
<