AngularJS渲染性能分析

做者:Jiang, Jilinjavascript


AngularJS中,經過數據綁定。可以十分方便的構建頁面。但是當面對複雜的循環嵌套結構時,渲染會遇到性能瓶頸。今天,咱們將經過一些列實驗,來測試AngularJS的渲染性能,對照ng-show。ng-if的使用場景。並對優化進行簡要分析。java

 

只是在此以前,咱們需要先簡單過一遍AngularJS相關的代碼:web

$apply: function(expr) {
  try {
    beginPhase('$apply');
    try {
      return this.$eval(expr);
    } finally {
      clearPhase();
    }
  } catch (e) {
    $exceptionHandler(e);
  } finally {
    try {
      $rootScope.$digest();
    } catch (e) {
      $exceptionHandler(e);
      throw e;
    }
  }
},

beginPhase和clearPhase用於對$rootScope.$$phase進行鎖定。假設發現反覆進入$apply階段則拋出異常。以避免出現死循環。express

$eval: function(expr, locals) {
  return $parse(expr)(this, locals);
},


 $parse調用的是$ParseProvider。數組

由於以後的實驗expr不傳值。因此$ParseProvider會直接返回空函數noop() {}。緩存

所以咱們就不作詳細的$ParseProvider內容分析了。app



在運行完$eval後。會調用$digest方法。異步

讓咱們看看$digest裏有些什麼:async

$digest: function() {
  var watch, value, last,
      watchers,
      length,
      dirty, ttl = TTL,
      next, current, target = this,
      watchLog = [],
      logIdx, logMsg, asyncTask;

  beginPhase('$digest');
  // Check for changes to browser url that happened in sync before the call to $digest
  $browser.$$checkUrlChange();

  if (this === $rootScope && applyAsyncId !== null) {
    // If this is the root scope, and $applyAsync has scheduled a deferred $apply(), then
    // cancel the scheduled $apply and flush the queue of expressions to be evaluated.
    $browser.defer.cancel(applyAsyncId);
    flushApplyAsync();
  }

  lastDirtyWatch = null;

  do { // "while dirty" loop
    dirty = false;
    current = target;

    while (asyncQueue.length) {
      try {
        asyncTask = asyncQueue.shift();
        asyncTask.scope.$eval(asyncTask.expression, asyncTask.locals);
      } catch (e) {
        $exceptionHandler(e);
      }
      lastDirtyWatch = null;
    }

    traverseScopesLoop:
    do { // "traverse the scopes" loop
      if ((watchers = current.$$watchers)) {
        // process our watches
        length = watchers.length;
        while (length--) {
          try {
            watch = watchers[length];
            // Most common watches are on primitives, in which case we can short
            // circuit it with === operator, only when === fails do we use .equals
            if (watch) {
              if ((value = watch.get(current)) !== (last = watch.last) &&
                  !(watch.eq
                      ?

equals(value, last) : (typeof value === 'number' && typeof last === 'number' && isNaN(value) && isNaN(last)))) { dirty = true; lastDirtyWatch = watch; watch.last = watch.eq ? copy(value, null) : value; watch.fn(value, ((last === initWatchVal) ? value : last), current); if (ttl < 5) { logIdx = 4 - ttl; if (!watchLog[logIdx]) watchLog[logIdx] = []; watchLog[logIdx].push({ msg: isFunction(watch.exp) ?ide

'fn: ' + (watch.exp.name || watch.exp.toString()) : watch.exp, newVal: value, oldVal: last }); } } else if (watch === lastDirtyWatch) { // If the most recently dirty watcher is now clean, short circuit since the remaining watchers // have already been tested. dirty = false; break traverseScopesLoop; } } } catch (e) { $exceptionHandler(e); } } } // Insanity Warning: scope depth-first traversal // yes, this code is a bit crazy, but it works and we have tests to prove it! // this piece should be kept in sync with the traversal in $broadcast if (!(next = ((current.$$watchersCount && current.$$childHead) || (current !== target && current.$$nextSibling)))) { while (current !== target && !(next = current.$$nextSibling)) { current = current.$parent; } } } while ((current = next)); // `break traverseScopesLoop;` takes us to here if ((dirty || asyncQueue.length) && !(ttl--)) { clearPhase(); throw $rootScopeMinErr('infdig', '{0} $digest() iterations reached. Aborting!\n' + 'Watchers fired in the last 5 iterations: {1}', TTL, watchLog); } } while (dirty || asyncQueue.length); clearPhase(); while (postDigestQueue.length) { try { postDigestQueue.shift()(); } catch (e) { $exceptionHandler(e); } } },


相同的,調用beginPhase改變階段。



$browser.$$checkUrlChange()用於檢測url是否變動。此次咱們也用不到:

function fireUrlChange() {
  if (lastBrowserUrl === self.url() && lastHistoryState === cachedState) {
    return;
  }

  lastBrowserUrl = self.url();
  lastHistoryState = cachedState;
  forEach(urlChangeListeners, function(listener) {
    listener(self.url(), cachedState);
  });
}

接着進行$rootScope和applyAsyncId推斷。假設是根Scope並且存在異步apply請求。則調用$eval並把隊列清空。也不是本次需要用到的部分。

 

進入循環,asyncQueue保存了$evalAsync方法的數據。

用不到。

 

以後設置了一個斷點,用於跳出內部循環:

traverseScopesLoop:

循環內推斷是否存在$$watchers列表,而後對watch單元進行變動匹配。每個頁面的數據綁定都會相應到一個watch單元。此處會檢查是否watch是深匹配,假設爲真會調用equals方法進行遞歸檢查,假設watch了一個巨大的對象。那麼equals會十分消耗性能。反之,則會檢查是不是NaN,js中NaN != NaN。然而假設原值和現值都是NaN,事實上是沒有變動過的。

if (watch) {
  if ((value = watch.get(current)) !== (last = watch.last) &&
      !(watch.eq
          ? equals(value, last)
          : (typeof value === 'number' && typeof last === 'number'
             && isNaN(value) && isNaN(last)))) {

假設循環後已經發現watch單元原值和現值相等,會跳出循環。

再次又一次驗證,目的是爲了防止某個watch調用回調函數後。使得以前的watch現值發生變化。

而當中也設置了ttl循環計數。以避免出現watch不斷改變產生死循環的問題。



接着,就是著名的crazy凝視了:

// Insanity Warning: scope depth-first traversal
// yes, this code is a bit crazy, but it works and we have tests to prove it!
// this piece should be kept in sync with the traversal in $broadcast

此處會深度優先遍歷,而後反覆上面的檢查。直到遍歷結束。

做者很是貼心的標註一下循環結束了:

// `break traverseScopesLoop;` takes us to here


後面的代碼就十分好懂了,clearPhase。而後處理DigestQueue結束循環。

 

以後檢查ttl數值,假設ttl值超出了10次(預設值),則會拋出過多循環的異常。

 

實驗

簡單的過了一遍代碼後。咱們開始作一下性能測試:(注:由於不一樣機器配置性能不一樣,渲染時間僅做橫向對照之用)

現在。若是咱們擁有2個用戶組。每組用戶擁有1000個用戶信息。用戶信息例如如下:

[{name: "user1"}, {name:"user2"},...]

 

咱們第一步作最簡單未通過優化的渲染:

<div>
   <div ng-repeat="user in userList">
      <label>Name</label>
      <p>{{user.name}}</p>
   </div>
</div>

切換分組渲染時間平均310ms左右。

 

track by

而後簡單使用優化track by優化:

ng-repeat="user in userList track by $index"

第一次渲染260ms左右。以後切換耗費11ms左右。

 

效果不錯。接着,咱們比較不一樣長度的數組切換比較。若是用戶組1長度仍然爲1000,用戶組2長度100:(下圖中,狀態一、2表明綁定數組的切換)

狀態1\狀態2

用戶組1

用戶組2

用戶組1

~0.3ms

~111ms

用戶組2

~175ms

~0.1ms

咱們可以看出,元素動態建立/刪除會極大影響渲染性能。

建立相同數量元素比刪除相同數量元素更消耗性能。

 

ng-show

基於以上實驗。咱們可以很是easy想到。假設咱們使用元素池,預先建立足量的元素。接着經過ng-show來動態調整顯示的元素。這樣性能是否會上升呢?

 

$scope.getTimes = function(n) {
return new Array(n);
};
<div ng-repeat="i in getTimes(1000) track by $index" ng-show="userList[$index]">
   <label>Name</label>
   <p>{{userList[$index].name}}</p>
</div>

狀態1\狀態2

用戶組1

用戶組2

用戶組1

~1.3ms

~42ms

用戶組2

~22ms

~1.0ms



可以發現。同組切換時間消耗少許添加。

但是相對的,異組切換性能大幅提高了。

這是由於web中,元素操做是十分消耗性能的操做。於是爲了性能。咱們需要儘量避免元素的建立/刪除。相同的,由於每次渲染,都會調用new Array和檢查ng-show屬性,從而致使了同組切換的時間添加了。




ng-if與ng-show

Angularjs中還有還有一個方法ng-if,它是僅僅有知足表達式條件纔會變動元素。對於用戶組切換,其毫無疑問會建立/刪除元素。只是在此,我仍是把數據羅列一下:

<div ng-repeat="i in getTimes(1000) track by $index" ng-if="userList[$index]">
   <label>Name</label>
   <p>{{userList[$index].name}}</p>
</div>

狀態1\狀態2

用戶組1

用戶組2

用戶組1

~11ms

~250ms

用戶組2

~300ms

~5.5ms


可以看出,使用緩存+ng-if。性能消耗會比本來沒有track by更消耗性能。


那麼ng-if的適用場景是什麼?是否所有的ng-if都適合被ng-show取代呢?讓咱們接下去繼續看看列子。

 

組合

首先。咱們對照一下有無緩存的初始化1000條數據的時間。

有緩存

無緩存

用戶組1

~276ms

~240ms

用戶組2

~278ms

~36ms

 

現在,咱們若是用戶有一個id屬性。UI中,依據id是除以5的餘數來作不一樣的渲染。規則例如如下:

餘數

渲染元素

0

畫一個2*2的table

1

顯示一個長度爲5的ul li列表

2

顯示一個checkbox的input

3

顯示一個textarea

4

顯示一個text input

你可能已經看出個人想法了,咱們的目的在於測試。假設存在多個不一樣渲染方式的狀況下,是否適合使用ng-show。咱們來看一下,(ng-switch近似ng-if,咱們一塊兒增長對照)

<div ng-repeat="user in userList track by $index">
   <label>Name</label>
   <p>{{user.name}}</p>

   <div ng-show="user.id % 5 === 0">
      <table>
         <tbody>
            <tr>
               <th>11</th>
               <th>12</th>
            </tr>
            <tr>
               <th>21</th>
               <th>22</th>
            </tr>
         </tbody>
      </table>
   </div>

   <div ng-show="user.id % 5 === 1">
      <ul>
         <li>1</li>
         <li>2</li>
         <li>3</li>
         <li>4</li>
         <li>5</li>
      </ul>
   </div>

   <div ng-show="user.id % 5 === 2">
      <input type="checkbox" />
   </div>

   <div ng-show="user.id % 5 === 3">
      <textarea></textarea>
   </div>

   <div ng-show="user.id % 5 === 4">
      <input type="text" />
   </div>
</div>
<div ng-repeat="user in userList track by $index">
   <label>Name</label>
   <p>{{user.name}}</p>

   <div ng-if="user.id % 5 === 0">
      <table>
         <tbody>
            <tr>
               <th>11</th>
               <th>12</th>
            </tr>
            <tr>
               <th>21</th>
               <th>22</th>
            </tr>
         </tbody>
      </table>
   </div>

   <div ng-if="user.id % 5 === 1">
      <ul>
         <li>1</li>
         <li>2</li>
         <li>3</li>
         <li>4</li>
         <li>5</li>
      </ul>
   </div>

   <div ng-if="user.id % 5 === 2">
      <input type="checkbox" />
   </div>

   <div ng-if="user.id % 5 === 3">
      <textarea></textarea>
   </div>

   <div ng-if="user.id % 5 === 4">
      <input type="text" />
   </div>
</div>


 

ng-show

ng-if

ng-switch

用戶組1

~557ms

~766ms

~858ms



接着,測試切換:

ng-show

ng-if

ng-switch

組1->組2

~260ms

~257ms

~261ms

組2->組1

~430ms

~470ms

~560ms

 好像ng-show各項數值都優於ng-if與ng-switch。只是還沒完,咱們繼續改動樣例。

爲用戶加入下面幾個屬性,相應綁定於以前定義的元素(m,n初始化時僞隨機生成以便於測試對照數值):

屬性

描寫敘述

matrix

一個m*n的數組

list

一個長度爲n的列表

desc

string

checked

boolean


<div ng-repeat="user in userList track by $index">
   <label>Name</label>
   <p>{{user.name}}</p>

   <div ng-show="user.id % 5 === 0">
      <table>
         <tbody>
            <tr ng-repeat="line in user.matrix track by $index">
               <th ng-repeat="val in line track by $index">{{val}}</th>
            </tr>
         </tbody>
      </table>
   </div>

   <div ng-show="user.id % 5 === 1">
      <ul>
         <li ng-repeat="val in user.list track by $index">{{val}}</li>
      </ul>
   </div>

   <div ng-show="user.id % 5 === 2">
      <input type="checkbox" ng-checked="user.checked" />
   </div>

   <div ng-show="user.id % 5 === 3">
      <textarea ng-model="user.desc"></textarea>
   </div>

   <div ng-show="user.id % 5 === 4">
      <input type="text" ng-model="user.desc" />
   </div>
</div>
<div ng-repeat="user in userList track by $index">
   <label>Name</label>
   <p>{{user.name}}</p>

   <div ng-if="user.id % 5 === 0">
      <table>
         <tbody>
            <tr ng-repeat="line in user.matrix track by $index">
               <th ng-repeat="val in line track by $index">{{val}}</th>
            </tr>
         </tbody>
      </table>
   </div>

   <div ng-if="user.id % 5 === 1">
      <ul>
         <li ng-repeat="val in user.list track by $index">{{val}}</li>
      </ul>
   </div>

   <div ng-if="user.id % 5 === 2">
      <input type="checkbox" ng-checked="user.checked" />
   </div>

   <div ng-if="user.id % 5 === 3">
      <textarea ng-model="user.desc"></textarea>
   </div>

   <div ng-if="user.id % 5 === 4">
      <input type="text" ng-model="user.desc" />
   </div>
</div>

 

ng-show

ng-if

ng-switch

用戶組1

~4678ms

~1800ms

~1990ms


是否是大吃一驚?緣由很是easy,由於ng-show僅僅是隱藏元素。

但是實際的數據綁定仍舊會被運行。

儘管在頁面上看不到,但是元素綁定的數據仍是一併更改了:



經過以上實驗,咱們很是easy分析出。當頁面佈局簡單時,可以經過ng-show+cachelist來實現高速的數據切換。而當元素組件存在大量元素變化的時候,使用ng-if/ng-switch來避免多餘的元素綁定。

經過二者結合的方式,可以使得程序在初始化和動態變化的時候保持更好的性能。相同的,在事件處理中。ng-if相較於ng-show會更有利於性能,但是假設事件綁定很少,使用ng-show則更佳。

相關文章
相關標籤/搜索