CSS3動畫卡頓效能優化解決方案

趁你還年輕發表於2018-01-30

CSS3 Transition

最近在開發小程式,與vue類似,它們都有生命週期這回事。

onLoad 監聽頁面載入 onReady 監聽頁面初次渲染完成 onShow 監聽頁面顯示

到底是什麼意思?

所以這又觸碰到了我的知識盲區,不過專案在磕磕絆絆中完成的差不多了,但是遇到了CSS3動畫渲染的效能問題,所以我也是被逼的,再回過頭來從瀏覽器渲染網頁的流程出發,去找動畫卡頓的癥結。

瀏覽器渲染網頁的流程如下:

使用 HTML 建立文件物件模型(DOM) 使用 CSS 建立 CSS 物件模型(CSSOM) 基於 DOM 和 CSSOM 執行指令碼(Scripts) 合併 DOM 和 CSSOM 形成渲染樹(Render Tree) 使用渲染樹佈局(Layout)所有元素 渲染(Paint)所有元素

可以結合Alon的這篇前端效能優化和安卓開發者選項的顯示頁面佈局。

安卓開發者選項的顯示頁面佈局

如何判斷手機app是native,webview還是hybird? 簡單說下,app中的一大塊是白色的沒有紅線標記出來的,但是上面有按鈕,圖片等時,就是webview,也就是通過一個偽瀏覽器去請求到的資料,斷網時開啟app沒有任何東西顯示在上面

小程式為hybird式開發

onLoad 監聽頁面載入 在渲染完介面之後,也就是通過.json中的配置項生成native介面後,開始渲染webview的部分,一個頁面只會呼叫一次。 onReady 監聽頁面初次渲染完成 一個頁面只會呼叫一次,代表頁面已經準備妥當,可以和檢視層進行互動。 onShow 監聽頁面顯示 每次開啟頁面都會去呼叫其中的函式。

我們的動畫應該放在哪裡?

應該放在onShow裡,因為這樣我每次開啟都能看到動畫。

為什麼會卡頓?

有一個前提必須要提,前端開發者們都知道,瀏覽器是單執行緒執行的。 但是我們要明確以下幾個概念:單執行緒,主執行緒和合成執行緒。

雖然說瀏覽器執行js是單執行緒執行(注意,是執行,並不是說瀏覽器只有1個執行緒,而是執行時,runing),但實際上瀏覽器的2個重要的執行執行緒,這 2 個執行緒協同工作來渲染一個網頁:主執行緒和合成執行緒。

一般情況下,主執行緒負責:執行 JavaScript;計算 HTML 元素的 CSS 樣式;頁面的佈局;將元素繪製到一個或多個點陣圖中;將這些點陣圖交給合成執行緒。

相應地,合成執行緒負責:通過 GPU 將點陣圖繪製到螢幕上;通知主執行緒更新頁面中可見或即將變成可見的部分的點陣圖;計算出頁面中哪部分是可見的;計算出當你在滾動頁面時哪部分是即將變成可見的;當你滾動頁面時將相應位置的元素移動到可視區域。

那麼為什麼會造成動畫卡頓呢?

原因就是主執行緒和合成執行緒的排程不合理。

下面來詳細說一下排程不合理的原因。

在使用height,width,margin,padding作為transition的值時,會造成瀏覽器主執行緒的工作量較重,例如從margin-left:-20px渲染到margin-left:0,主執行緒需要計算樣式margin-left:-19px,margin-left:-18px,一直到margin-left:0,而且每一次主執行緒計算樣式後,合成程式都需要繪製到GPU然後再渲染到螢幕上,前後總共進行20次主執行緒渲染,20次合成執行緒渲染,20+20次,總計40次計算。

主執行緒的渲染流程,可以參考瀏覽器渲染網頁的流程:

使用 HTML 建立文件物件模型(DOM) 使用 CSS 建立 CSS 物件模型(CSSOM) 基於 DOM 和 CSSOM 執行指令碼(Scripts) 合併 DOM 和 CSSOM 形成渲染樹(Render Tree) 使用渲染樹佈局(Layout)所有元素 渲染(Paint)所有元素

也就是說,主執行緒每次都需要執行Scripts,Render Tree ,Layout和Paint這四個階段的計算。

而如果使用transform的話,例如tranform:translate(-20px,0)到transform:translate(0,0),主執行緒只需要進行一次tranform:translate(-20px,0)到transform:translate(0,0),然後合成執行緒去一次將-20px轉換到0px,這樣的話,總計1+20計算。

可能會有人說,這才提升了19次,有什麼好效能提升的?

假設一次10ms。

那麼就減少了約190ms的耗時。

會有人說,辣雞,才190ms,無所謂。

那麼如果margin-left是從-200px到0呢,一次10ms,10ms*199≈2s。

還會有人說,辣雞,也就2s,無所謂。

你忘了單執行緒這回事了嗎?

如果網頁有3個動畫,3*2s=6s,就是6s的效能提升。 由於資料是猜測的,所以暫時不考慮其真實性,文章後面我使用chrome devtools的performance做了一個實驗。

要知道,在"客戶至上"的今天,好的使用者體驗是所有產品的必須遵守的一條規則,無論是對於開發者還是產品經理,追求極致的效能都是我們打造一個好的產品所必備的品質。

可能看了我的略不專業的分析後,大家對主執行緒,合成執行緒以及它們在2種效能不同動畫方案上的工作流程還不是很瞭解,可以去看一篇翻譯過來的部落格(英文原版連結已經失效了):深入瀏覽器理解CSS animations 和 transitions的效能問題

這篇文章完美講述了瀏覽器主執行緒和合成執行緒的區別,並且舉了一個高度從100px變化到200px的2種動畫方案的對比,對主執行緒和合成執行緒的整個工作流程做了很詳盡的講解,真心建議認真閱讀一遍。

回過頭來總結下,css3動畫卡頓的解決方案: 在使用css3 transtion做動畫效果時,優先選擇transform,儘量不要使用height,width,margin和padding。

transform為我們提供了豐富的api,例如scale,translate,rotate等等,但是在使用時需要考慮相容性。但其實對於大多數css3來說,mobile端支援性較好,desktop端支援性需要格外注意。


補充:為了增強本文的說服力,特地回家做了一個實驗,程式碼如下。

<!DOCTYPE html>
<html>
<head>
  <meta charset="utf-8" />
  <meta http-equiv="X-UA-Compatible" content="IE=edge">
  <title>Page Title</title>
  <meta name="viewport" content="width=device-width, initial-scale=1">
  <style>
    .margin-transition{
      /* margin-left: 0; */
      background: rgba(0,0,255,0.3);
      transition: margin-left 1s;
    }
    .transform-transition{
      /* transform: translate(0,0); */
      background: rgba(0,255,0,0.3);
      transition: transform 1s;
    }
    .common{
      height: 300px;
      width: 300px;
    }
  </style>
</head>
<body>
  <div class="margin-transition common" id="marginTransition">
    <p>transition:margin-left 1s</p>
  </div>
  <div class="transform-transition common" id="transformTransition">
      <p>transition:tranform 1s</p>
  </div>
  <button id="control">見證奇蹟</button>
  <script>
      var btn = document.getElementById('control');
      var marginTransition = document.getElementById('marginTransition');
      var transformTransition = document.getElementById('transformTransition');
      btn.addEventListener("click",function(){
        console.log(marginTransition.style,transformTransition.style)
        marginTransition.style.marginLeft = "500px";
        transformTransition.style.transform = "translate(500px,0)"
      })
  </script>  
</body>
</html>
複製程式碼

我將主要藉助chrome devtools的performance工具對比二者的效能差異。 先來看margin動畫,動態修改DOM節點的margin-left值從0到500px;。

transition: margin-left 1s;
複製程式碼

margin動畫實驗
margin動畫總耗時
margin動畫GPU使用率

再來看下transform動畫,動態修改DOM節點的transform值從translate(0,0)到translate(500px,0)。

transition: transform 1s;
複製程式碼

transform動畫實驗

transform動畫總耗時

transform動畫GPU使用率

可能圖片不是很好地能說明效能差異,那麼我們來列一張耗時對比表,方便我們計算。

耗時 margin transform
Summery 3518ms 2286ms
Scripting 1.8ms 2.9ms
Rendering 22.5ms 6.9ms
Painting 9.7ms 1.6ms
Other 39.3ms 25.2ms
Idle( browser is waiting on the CPU or GPU to do some processing) 3444.4ms 2249.8ms
GPU使用率 4.1MB 1.7MB

通過上表我們可以計算出明margin,transform與transition組合實現CSS3動畫效果時的效能差異引數。

關鍵效能引數 margin transform
實際動畫耗時(總時間 減去 空閒時間) 73.6ms 36.2ms

計算得出,transform動畫耗時約等於margin動畫耗時的0.49倍,效能優化50%。

由於我對Other的所做的具體事情不是很清楚,所以這裡的實際動畫時間也有可能還要減掉Other中的時間,下表是我們減掉後的資料。

關鍵效能引數 margin transform
實際動畫耗時(總時間 減去 其他時間和空閒時間) 34.3ms 11ms

計算得出,transform動畫耗時約等於margin動畫耗時的0.32倍,效能優化接近70%。

也就是說,無論我們減去還是不減去Other的時間,我們採用transform實現動畫的方式都比margin動畫快。

不精確的得出一個小結論:transform比margin效能好50%~70%

雖然會有50%~70%的效能提升,但是需要注意硬體差異,硬體好的情況下可能不能發現卡頓或者其他的一些效能上的問題。 例如在開發小程式的過程中,模擬器是位於desktop端的,因此它的硬體效能效能更好,例如CPU,GPU。但是一旦在mobile端執行,例如ios或者android上執行時,就可能會出現效能問題,這就是因為移動端的硬體條件遜於PC端導致的。

所以說,效能問題是一直存在的,只不過硬體差異會導致效能影響的程度不同。

所以我們再次回過頭來,總結出css3動畫卡頓的解決方案: 在使用css3 transtion做動畫效果時,優先選擇transform,儘量不要使用height,width,margin和padding。

That's it !


2018年11月28日更新 為什麼transform更加平滑?

合成器的優點在於,其工作無關主執行緒,合成器執行緒不需要等待樣式計算或者 JS 執行,這就是為什麼合成器相關的動畫 最流暢,如果某個動畫涉及到佈局或者繪製的調整,就會涉及到主執行緒的重新計算,自然會慢很多。

引用自infoQ的好文:史上最全!圖解瀏覽器的工作原理

與我們的部落格上下文結合起來,解釋如下: 在使用css3 transtion做動畫效果時,transform實現的動畫是與合成器執行緒相關的,不需要等待主執行緒樣式計算或者 JS 執行,計算速度是很快的;而使用height,width,margin和padding時,導致佈局和繪製的調整,主執行緒需要重新計算樣式並且執行JS,計算速度自然就慢了。

參考: sy-tang.github.io/2014/05/14/… jinlong.github.io/2017/05/08/… blog.csdn.net/yeana1/arti… www.jianshu.com/p/b70b72de3… developers.google.com/web/tools/c… blogs.adobe.com/webplatform…

相關文章