做者: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 |
<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則更佳。