国产xxxx99真实实拍_久久不雅视频_高清韩国a级特黄毛片_嗯老师别我我受不了了小说

資訊專欄INFORMATION COLUMN

Angular 1 深度解析:臟數(shù)據(jù)檢查與 angular 性能優(yōu)化

fasss / 715人閱讀

摘要:通常寫代碼時我們無需主動調(diào)用或是因為在外部對我們的回調(diào)函數(shù)做了包裝。類似的不只是這些事件回調(diào)函數(shù),還有等。常量依舊會重復檢查。會檢查中有沒有一個名為的成員。

TL;DR

臟檢查是一種模型到視圖的數(shù)據(jù)映射機制,由 $apply$digest 觸發(fā)。

臟檢查的范圍是整個頁面,不受區(qū)域或組件劃分影響

使用盡量簡單的綁定表達式提升臟檢查執(zhí)行速度

盡量減少頁面上綁定表達式的個數(shù)(單次綁定和ng-if

ng-repeat 添加 track by 讓 angular 復用已有元素

什么是臟數(shù)據(jù)檢查(Dirty checking)

Angular 是一個 MVVM 前端框架,提供了雙向數(shù)據(jù)綁定。所謂雙向數(shù)據(jù)綁定(Two-way data binding)就是頁面元素變化會觸發(fā) View-model 中對應數(shù)據(jù)改變,反過來 View-model 中數(shù)據(jù)變化也會引發(fā)所綁定的 UI 元素數(shù)據(jù)更新。操作數(shù)據(jù)就等同于操作 UI。

看似簡單,其實水很深。UI 元素變化引發(fā) Model 中數(shù)據(jù)變化這個通過綁定對應 DOM 事件(例如 inputchange)可以簡單的實現(xiàn);然而反過來就不是那么容易。

比如有如下代碼:

用戶點擊了 button,angular 執(zhí)行了一個叫 onClick 的方法。這個 onClick 的方法體對于 angular 來說是黑盒,它到底做了什么不知道。可能改了 $scope.content1 的值,可能改了 $scope.content2 的值,也可能兩個值都改了,也可能都沒改。

那么 angular 到底應該怎樣得知 onClick() 這段代碼后是否應該刷新 UI,應該更新哪個 DOM 元素?

angular 必須去挨個檢查這些元素對應綁定表達式的值是否有被改變。這就是臟數(shù)據(jù)檢查的由來(臟數(shù)據(jù)檢查以下簡稱臟檢查)。

臟檢查如何被觸發(fā)

angular 會在可能觸發(fā) UI 變更的時候進行臟檢查:這句話并不準確。實際上,臟檢查是 $digest](https://docs.angularjs.org/api/ng/type/$rootScope.Scope#$digest) 執(zhí)行的,另一個更常用的用于觸發(fā)臟檢查的函數(shù) [$apply 其實就是 $digest 的一個簡單封裝(還做了一些抓異常的工作)。

通常寫代碼時我們無需主動調(diào)用 $apply$digest 是因為 angular 在外部對我們的回調(diào)函數(shù)做了包裝。例如常用的 ng-click,這是一個指令(Directive),內(nèi)部實現(xiàn)則 類似

DOM.addEventListener("click", function ($scope) {
  $scope.$apply(() => userCode());
});

可以看到:ng-click 幫我們做了 $apply 這個操作。類似的不只是這些事件回調(diào)函數(shù),還有 $http$timeout 等。我聽很多人抱怨說 angular 這個庫太大了什么都管,其實你可以不用它自帶的這些服務(Service),只要你記得手工調(diào)用 $scope.$apply

臟檢查的范圍

前面說到:angular 會對所有綁定到 UI 上的表達式做臟檢查。其實,在 angular 實現(xiàn)內(nèi)部,所有綁定表達式都被轉(zhuǎn)換為 $scope.$watch()。每個 $watch 記錄了上一次表達式的值。有 ng-bind="a" 即有 $scope.$watch("a", callback),而 $scope.$watch 可不會管被 watch 的表達式是否跟觸發(fā)臟檢查的事件有關(guān)。

例如:

問:點擊 TEST 這個按鈕時會觸發(fā)臟檢查嗎?觸發(fā)幾次?

首先:ng-click="" 什么都沒有做。angular 會因為這個事件回調(diào)函數(shù)什么都沒做就不進行臟檢查嗎?不會。

然后:#span1 被隱藏掉了,會檢查綁定在它上面的表達式嗎?盡管用戶看不到,但是 $scope.$watch("content", callback) 還在。就算你直接把這個 span 元素干掉,只要 watch 表達式還在,要檢查的還會檢查。

再次:重復的表達式會重復檢查嗎?會。

最后:別忘了 ng-show="false"。可能是因為 angular 的開發(fā)人員認為這種綁定常量的情況并不多見,所以 $watch 并沒有識別所監(jiān)視的表達式是否是常量。常量依舊會重復檢查。

所以:

答:觸發(fā)三次。一次 false,一次 content,一次 content

所以說一個綁定表達式只要放在當前 DOM 樹里就會被監(jiān)視,不管它是否可見,不管它是否被放在另一個 Tab 里,更不管它是否與用戶操作相關(guān)。

另外,就算在不同 Controller 里構(gòu)造的 $scope 也會互相影響,別忘了 angular 還有全局的 $rootScope,你還可以 $scope.$emit。angular 無法保證你絕對不會在一個 controller 里更改另一個 controller 生成的 scope,包括 自定義指令(Directive)生成的 scopeAngular 1.5 里新引入的組件(Component)。

所以說不要懷疑用戶在輸入表單時 angular 會不會監(jiān)聽頁面左邊導航欄的變化。

臟檢查與運行效率

臟檢查慢嗎?

說實話臟檢查效率是不高,但是也談不上有多慢。簡單的數(shù)字或字符串比較能有多慢呢?十幾個表達式的臟檢查可以直接忽略不計;上百個也可以接受;成百上千個就有很大問題了。綁定大量表達式時請注意所綁定的表達式效率。建議注意一下幾點:

表達式(以及表達式所調(diào)用的函數(shù))中少寫太過復雜的邏輯

不要連接太長的 filter(往往 filter 里都會遍歷并且生成新數(shù)組)

不要訪問 DOM 元素。

使用單次綁定減少綁定表達式數(shù)量

單次綁定(One-time binding 是 Angular 1.3 就引入的一種特殊的表達式,它以 :: 開頭,當臟檢查發(fā)現(xiàn)這種表達式的值不為 undefined 時就認為此表達式已經(jīng)穩(wěn)定,并取消對此表達式的監(jiān)視。這是一種行之有效的減少綁定表達式數(shù)量的方法,與 ng-repeat 連用效果更佳(下文會提到),但過度使用也容易引發(fā) bug。

善用 ng-if 減少綁定表達式的數(shù)量

如果你認為 ng-if 就是另一種用于隱藏、顯示 DOM 元素的方法你就大錯特錯了。

ng-if 不僅可以減少 DOM 樹中元素的數(shù)量(而非像 ng-hide 那樣僅僅只是加個 display: none),每一個 ng-if 擁有自己的 scopeng-if 下面的 $watch 表達式都是注冊在 ng-if 自己 scope 中。當 ng-if 變?yōu)?falseng-if 下的 scope 被銷毀,注冊在這個 scope 里的綁定表達式也就隨之銷毀了。

考慮這種 Tab 選項卡實現(xiàn):

  • Tab 1 title
  • Tab 2 title
  • Tab 3 title
  • Tab 4 title
[[Tab 1 body...]]
[[Tab 2 body...]]
[[Tab 3 body...]]
[[Tab 4 body...]]

對于這種會反復隱藏、顯示的元素,通常人們第一反應都是使用 ng-showng-hide 簡單的用 display: none 把元素設(shè)置為不可見。

然而入上文所說,肉眼不可見不代表不會跑臟檢查。如果將 ng-show 替換為 ng-ifng-switch-when

[[Tab 1 body...]]
[[Tab 2 body...]]
[[Tab 3 body...]]
[[Tab 4 body...]]

有如下優(yōu)點:

首先 DOM 樹中的元素個數(shù)顯著減少至四分之一,降低內(nèi)存占用

其次 $watch 表達式也減少至四分之一,提升臟檢查循環(huán)的速度

如果這個 tab 下面有 controller(例如每個 tab 都被封裝為一個組件),那么僅當這個 tab 被選中時該 controller 才會執(zhí)行,可以減少各頁面的互相干擾

如果 controller 中調(diào)用接口獲取數(shù)據(jù),那么僅當對應 tab 被選中時才會加載,避免網(wǎng)絡(luò)擁擠

當然也有缺點:

DOM 重建本身費時間

如果 tab 下有 controller,那么每次該 tab 被選中時 controller 都會被執(zhí)行

如果在 controller 里面調(diào)接口獲取數(shù)據(jù),那么每次該 tab 被選中時都會重新加載

各位讀者自己取舍。

當臟檢查遇上數(shù)組

ng-repeat!這就更有(e)趣(xin)了。通常的綁定只是去監(jiān)聽一個值的變化(綁定對象也是綁定到對象里的某個成員),而 ng-repeat 卻要監(jiān)視一整個數(shù)組對象的變化。例如有:

會生成 4 個 li 元素

1

2

3

4

沒有問題。如果我添加一個按鈕如下:

請考慮:當用戶點擊這個按鈕會發(fā)生什么?

我們一步一步分析。開始的時候,angular 記錄了 array 的初始狀態(tài)為:

[
  { "value": 1 },
  { "value": 2 },
  { "value": 3 },
  { "value": 4 }
]

當用戶點擊按鈕后,數(shù)組的第一個元素被刪除了,array 變?yōu)椋?/p>

[
  { "value": 2 },
  { "value": 3 },
  { "value": 4 }
]

兩者比較:

array.length = 4 => array.length = 3

array[0].value = 1 => array[0].value = 2

array[1].value = 2 => array[1].value = 3

array[2].value = 3 => array[2].value = 4

array[3].value = 4 => array[3].value = undefinedarray[4]undefined,則 undefined.value 為 undefined,見 Angular 表達式的說明)

如同你所見:angular 經(jīng)過比較,看到的是:

數(shù)組長度減少了 1

數(shù)組第 1 個元素的 value 被改為 2

數(shù)組第 2 個元素的 value 被改為 3

數(shù)組第 3 個元素的 value 被改為 4

反應到 DOM 元素上就是:

第 1 個 li 內(nèi)容改為 2

第 2 個 li 內(nèi)容改為 3

第 3 個 li 內(nèi)容改為 4

第 4 個 li 刪掉

可以看到,刪除一個元素導致了整個 ul 序列的刷新。要知道 DOM 操作要比 JS 變量操作要慢得多,類似這樣的無用操作最好能想辦法避免。

那么問題出在哪里呢?用戶刪除了數(shù)組的第一個元素,導致了整個數(shù)組元素前移;然而 angular 沒法得知用戶做了這樣一個刪除操作,只能傻傻的按下標一個一個比。

那么只要引入一種機制來標記數(shù)組的每一項就好了吧。于是 angular 引入了 track by

詳解 track by

用來標記數(shù)組元素的一定是數(shù)組里類似 ID 的某個值。這個值一定要符合以下這兩個特點。

不能重復。ID 重復了什么鬼

值一定要簡單。ID 是用于比較相等的,有時候由于算法不同可能還要比較大小,處于速度考慮不能太復雜。

基于這兩個特點。如果用戶沒有給 ng-repeat 指定 track by 的表達式,則默認為內(nèi)置函數(shù) $id。$id 會檢查 item 中有沒有一個名為 $$hashKey` 的成員。如有,返回其值;如沒有,則生成一個新的唯一值寫入。這就是數(shù)組中那個奇怪的 `$$hashKey 成員來歷,默認值是 "object:X"(你問我為什么是個字符串而不是數(shù)字?我怎么知道。。。)

還是前面的問題,引入 track by 后再來看。因為沒有指定 track by,則默認為 $id(item),實際為 $$hashKey

開始的時候,$id(item) 給數(shù)組中所有項創(chuàng)建了 $$hashKey

這時 angular 記錄了 array 的初始狀態(tài)為:

[
  { "value": 1, "$$hashKey": "object:1" },
  { "value": 2, "$$hashKey": "object:2" },
  { "value": 3, "$$hashKey": "object:3" },
  { "value": 4, "$$hashKey": "object:4" }
]

當用戶點擊按鈕后,數(shù)組的第一個元素被刪除了,array 變?yōu)椋?/p>

[
  { "value": 2, "$$hashKey": "object:2" },
  { "value": 3, "$$hashKey": "object:3" },
  { "value": 4, "$$hashKey": "object:4" }
]

先比較 track by 的元素,這里為 $id(item),即 $$hashKey

"object:1" => "object:2"

"object:2" => "object:3"

"object:3" => "object:4"

"object:4" => undefined

兩者對不上,說明數(shù)組被做了增刪元素或者移動元素的操作。將其規(guī)整

"object:1" => undefined

"object:2" => "object:2"

"object:3" => "object:3"

"object:4" => "object:4"

那么顯然,第一個元素被刪除了。再比較剩余的元素

array[0].value = 2 => array[0].value = 2

array[1].value = 3 => array[1].value = 3

array[2].value = 4 => array[2].value = 4

結(jié)論是:

原數(shù)組第一個元素被刪除

其他沒變

angular 通過將新舊數(shù)組的 track by 元素做 diff 猜測用戶的行為,最大可能的減少 DOM 樹的操作,這就是 track by 的用處。

默認 track by 的坑

So far so good! 然而需求某天有變,程序員小哥決定用 filter 給數(shù)組做 map 后再渲染。

map 定義如下:

xxModule.filter("map", function () {
  return arr => arr.map(item => ({ value: item.value + 1 }));
});

ng-repeat 執(zhí)行時先計算表達式 array | myMap 的值:

arrayForNgRepeat = [
  { value: 2 },
  { value: 3 },
  { value: 4 },
  { value: 5 },
]

注意數(shù)組 arrayForNgRepeat 和原來的數(shù)組 array 不是同一個引用,因為 filter 里的 map 操作生成了一個新數(shù)組,每一項都是新對象,跟原數(shù)組無關(guān)。

ng-repeat 時,angular 發(fā)現(xiàn)用戶沒有指定 track by,按照默認邏輯,使用 $id(item) 作為 track by,添加 $$hashKey

arrayForNgRepeat = [
  { value: 2, "$$hashKey": "object:1" },
  { value: 3, "$$hashKey": "object:2" },
  { value: 4, "$$hashKey": "object:3" },
  { value: 5, "$$hashKey": "object:4" },
]

生成 DOM:

2

3

4

5

這里請再次注意:數(shù)組 arrayForNgRepeat 與原始數(shù)組 array 沒有任何關(guān)系,數(shù)組本身是不同的引用,數(shù)組里的每一項也是不同引用。修改新數(shù)組的成員不會影響到原來的數(shù)組。

這時 array 的值:

array = [
  { value: 1 },
  { value: 2 },
  { value: 3 },
  { value: 4 },
]

這時用戶的某個無關(guān)操作觸發(fā)了臟檢查。針對 ng-repeat 表達式,首先計算 array | myMap 的值:

newArrayForNgRepeat = [
  { value: 2 },
  { value: 3 },
  { value: 4 },
  { value: 5 },
]

先比較 track by 的元素。用戶沒有指定,默認為 $id(item)

$id 發(fā)現(xiàn)數(shù)組中有一些元素沒有 $$hashKey`,則給它們填充新 `$$hashKey,結(jié)果為

newArrayForNgRepeat = [
  { value: 2, "$$hashKey": "object:5" },
  { value: 3, "$$hashKey": "object:6" },
  { value: 4, "$$hashKey": "object:7" },
  { value: 5, "$$hashKey": "object:8" },
]

這時兩邊的 track by 的實際結(jié)果為

"object:1" => "object:5"

"object:2" => "object:6"

"object:3" => "object:7"

"object:4" => "object:8"

兩者對不上,說明數(shù)組被做了增刪元素或者移動元素的操作。將其規(guī)整

"object:1" => undefined

"object:2" => undefined

"object:3" => undefined

"object:4" => undefined

undefined => "object:5"

undefined => "object:6"

undefined => "object:7"

undefined => "object:8"

結(jié)論是:

原數(shù)組全部 4 個元素被刪除

新添加了 4 個元素

于是 angular 把原來所有 li 刪除,再創(chuàng)建 4 個新的 li 元素,填充它們的 textContent,放到 ul

如果懷疑我說的話,請自己在瀏覽器里測試。你可以清楚的看到調(diào)試工具里 DOM 樹的閃爍

track by 與性能

不恰當?shù)?ng-repeat 會造成 DOM 樹反復重新構(gòu)造,拖慢瀏覽器響應速度,造成頁面閃爍。除了上面這種比較極端的情況,如果一個列表頻繁拉取 Server 端數(shù)據(jù)自刷新的話也一定要手工添加 track by,因為接口給前端的數(shù)據(jù)是不可能包含 $$hashKey 這種東西的,于是結(jié)果就造成列表頻繁的重建。

其實不必考慮那么多,總之加上沒壞處,至少可以避免 angular 生成 $$hashKey 這種奇奇怪怪的東西。所以

請給 ng-repeat 手工添加 track by

重要的事情再說一遍

請給 ng-repeat 手工添加 track by

通常列表都是請求接口從數(shù)據(jù)庫中讀取返回的。通常數(shù)據(jù)庫中的記錄都有一個 id 字段做主鍵,那么這時使用 id 作為 track by 的字段是最佳選擇。如果沒有,可以選擇一些業(yè)務字段但是確保不會重復的。例如一個連表頭都是動態(tài)生成的表格,表頭就可以使用其字段名作為 track by 的字段(對象的 key 是不會重復的)。

如果真的找不到用于 track by 的字段,讓 angular 自動生成 $$hashKey 也不是不可以,但是切記檢查有沒有出現(xiàn) DOM 元素不斷重刷的現(xiàn)象,除了仔細看調(diào)試工具的 DOM 樹是否閃爍之外,給列表中的元素添加一個特別的標記(比如 style="background: red"),也是一個行之有效的方法(如果這個標記被意外清除,說明原來的 DOM 元素被刪除了)。

除非真的沒辦法,不推薦使用 $index 作為 track by 的字段。

track by 與 單次綁定 連用

track by 只是讓 angular 復用已有 DOM 元素。數(shù)組每個子元素內(nèi)部綁定表達式的臟檢查還是免不了的。然而對于實際應用場景,往往是數(shù)組整體改變(例如分頁),數(shù)組每一項通常卻不會多帶帶變化。這時就可以通過使用單次綁定大量減少 $watch 表達式的數(shù)量。例如

  • a:
    b:
    c:
    d:
    e:
  • 除非 track by 字段改變造成的 DOM 樹重建,item.a 等一旦顯示在頁面上后就不會再被監(jiān)視。

    如果每行有 5 個綁定表達式,每頁顯示 20 條記錄,通過這種方法每頁就可以減少 5 * 20 = 100 個綁定表達式的監(jiān)視。

    注意:如果在 ng-repeat 內(nèi)部使用的單次綁定,就一定不要用 track by $index。否則用戶切換下一頁頁面也不會更新。

    使用分頁減少綁定個數(shù)

    這個就不多說了。能后端分頁的就后端分頁;接口不支持分頁的也要前端分頁;前端分頁時可以簡單的寫個 filterArray.prototype.slice 實現(xiàn)。

    能直接減少數(shù)組中項的個數(shù)就不要在 ng-repeat 中每項上寫 ng-showng-if

    寫在最后的話

    臟檢查這個東西,其實在三大主流前端框架中或多或少都有涉及。React 每次生成新的 Virtual DOM,與舊 Virtual DOM 的 diff 操作本來就可以看做一次臟檢查。Vue 從相對徹底的拋棄了臟檢查機制,使用 Property 主動觸發(fā) UI 更新,但是 Vue 仍然不能拋棄 track by 這個東西。

    既然臟檢查在三大主流框架里或多或少都有所保留,為什么唯獨 Angular 的性能被廣為詬病呢?其實還是說在 Angular 1 的機制下,臟檢查的執(zhí)行范圍過大以及頻率太過頻繁了。Angular 1.5 從 Angular 2+ 引入了組件(Component)的概念,然而形似而神非,其實只是一個特殊的 Directive 馬甲而已,并不能將臟檢查的執(zhí)行范圍限制在各個組件之內(nèi),所以并不能本質(zhì)的改變 Angular 1 臟檢查機制效率低下的現(xiàn)狀。

    也許 Angular 1 終將被淘汰。但 Angular 作為前端第一個 MVVM 框架,著實引發(fā)了前端框架更新?lián)Q代的熱潮。百足之蟲死而不僵,不管怎么樣我還得繼續(xù)維護停留在電腦里的 Angular 1 項目。不過也許老板哪天大發(fā)慈悲給我們用 Vue 重構(gòu)整個項目的時間,將來的事情誰知道呢?

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

    轉(zhuǎn)載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/51072.html

    相關(guān)文章

    • Angular 1 深度解析數(shù)據(jù)檢查 angular 性能優(yōu)化

      摘要:通常寫代碼時我們無需主動調(diào)用或是因為在外部對我們的回調(diào)函數(shù)做了包裝。類似的不只是這些事件回調(diào)函數(shù),還有等。常量依舊會重復檢查。會檢查中有沒有一個名為的成員。 TL;DR 臟檢查是一種模型到視圖的數(shù)據(jù)映射機制,由 $apply 或 $digest 觸發(fā)。 臟檢查的范圍是整個頁面,不受區(qū)域或組件劃分影響 使用盡量簡單的綁定表達式提升臟檢查執(zhí)行速度 盡量減少頁面上綁定表達式的個數(shù)(單次綁定...

      VioletJack 評論0 收藏0
    • angular性能優(yōu)化心得

      摘要:本文針對的讀者具備性能優(yōu)化的相關(guān)知識雅虎條性能優(yōu)化原則高性能網(wǎng)站建設(shè)指南等擁有實戰(zhàn)經(jīng)驗。這種機制能減少瀏覽器次數(shù),從而提高性能。僅會檢查該和它的子,當你確定當前操作僅影響它們時,用可以稍微提升性能。 搬運自: http://atian25.github.io/2014/05/09/angular-performace/ 不知不覺,在項目中用angular已經(jīng)半年多了,踩了很多坑...

      guqiu 評論0 收藏0
    • 尋找真兇Echarts or Angular

      摘要:我們再來看一下調(diào)用棧,如下圖從圖中我們發(fā)現(xiàn)了一個調(diào)用棧的代碼執(zhí)行過,還記得里提到嗎發(fā)起臟檢查的通知者,它代理了原生事件,任何一個原生異步事件的觸發(fā)都會導致的運行。 尋找真兇Echarts or Angular 這是一篇故事,就如同技術(shù),我們所追求的不是一個結(jié)局,而是那些深受啟發(fā)與共鳴的過程,那是我們成長的經(jīng)驗與生產(chǎn)力的積淀! 故事開始于瘋了的ionic3應用 頁面打開,什么也沒做5s里...

      sumory 評論0 收藏0
    • Immutable & Redux in Angular Way

      摘要:來源于社區(qū),時至今日已經(jīng)基本成為的標配了。部分很簡單,要根據(jù)傳入的執(zhí)行不同的操作。當性能遇到瓶頸時基本不會遇到,可以更改,保證傳入數(shù)據(jù)來提升性能。當不再能滿足程序開發(fā)的要求時,可以嘗試使用進行函數(shù)式編程。 Immutable & Redux in Angular Way 寫在前面 AngularJS 1.x版本作為上一代MVVM的框架取得了巨大的成功,現(xiàn)在一提到Angular,哪怕是已...

      lunaticf 評論0 收藏0

    發(fā)表評論

    0條評論

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