Angularjs中的$apply及優(yōu)化使用詳解
前言
對于一個(gè)在前端屬于純新手的我來說,Javascript都還是一知半解,要想直接上手angular JS,遇到的阻力還真是不少。不過我相信,只要下功夫,即使是反人類的設(shè)計(jì)也不是什么大的問題。
今天,我們要聊得是Angularjs中的小明星$apply。當(dāng)我們數(shù)據(jù)更新了,但是view層卻沒反應(yīng)時(shí),總能聽到有人說,用apply吧,然后,懵懂無知的我們,在賦值代碼后面加了$scope.$apply()
,然后就驚喜的發(fā)現(xiàn)。噢,真的更新了。
然而,有些時(shí)候,編譯器會(huì)無情的給你返回
Error: $digest already in progress
那么,導(dǎo)致這些現(xiàn)象的原因時(shí)什么的呢?$apply究竟干了啥?聽我慢慢到來。
一.$apply的作用
$apply()函數(shù)可以從Angular框架的外部讓表達(dá)式在Angular上下文內(nèi)部執(zhí)行。
上面是AngularJs權(quán)威教程中的一句話。什么意思呢?
首先,你要清楚,在原生js或者第三方框架下,修改model,是有可能不會(huì)觸發(fā)視圖更新的,比如setTimeout、jquery插件。為什么?因?yàn)樗麄兠撾x了Angularjs的上下文,Angularjs并不能監(jiān)聽到數(shù)據(jù)的改變??蠢?。
1.setTimeout
html:
<p>{{name}}</p>
js:
$scope.name="張三"; setTimeout(function(){ $scope.name = '李四'; //$scope.$apply() },500)
首先,name等于張三,500ms后,我把他賦值為李四,但是,頁面上并沒有改變,依然是張三。
而,我們把$scope.$apply()
放開,就正常了,張三成功變?yōu)槔钏摹?/p>
2.第三方插件
html:
<p>Date: <input type="text" id="datepicker"></p> <p> <header>所選日期</header> {{selectedDate}} </p>
js:
$scope.selectedDate = ''; $( function() { $( "#datepicker" ).datepicker({ onClose: function( selectedDate ) { $scope.selectedDate = selectedDate; // $scope.$apply(); } }); } );
這是jquery的datepicker插件,當(dāng)我們選定日期后,下面的日期應(yīng)該隨之顯現(xiàn),而現(xiàn)在卻沒有。這種情況就必須依靠$apply(),才能更新視圖。
以上兩種情況,都因?yàn)椴惶幱贏ngularjs上下文中,導(dǎo)致監(jiān)聽不到數(shù)據(jù)的變化。而$apply究竟干了什么,才導(dǎo)致數(shù)據(jù)更新正常了呢?
其實(shí)$apply相當(dāng)于一個(gè)觸發(fā)器,它的作用就是觸發(fā)digest循環(huán),從而更新視圖。
在digest是Angularjs的核心,是它實(shí)現(xiàn)了神奇的數(shù)據(jù)綁定。凡是觸發(fā)事件,必會(huì)觸發(fā)digest循環(huán),比如,我們數(shù)值的ng事件,click啊,change,實(shí)際上都是觸發(fā)了digest循環(huán)。
所以,我們所做的事,其實(shí)就是手動(dòng)觸發(fā)了digest循環(huán)。關(guān)于digest循環(huán),屬于題外話,這里不做過多介紹,想深入了解的同學(xué),可以看看書籍,或者百度。
二.更好地運(yùn)用digest循環(huán)
在Angularjs中,除了$apply可以觸發(fā)digest循環(huán)外,還有其他的方法,也可以觸發(fā)此循環(huán)。而且$apply往往時(shí)最壞的選擇。下面推薦一些更好的選擇。
1.$digest
$scope.$digest()
的速度要比$apply要快,因?yàn)樗桓庐?dāng)前作用域和子作用域的值,對于父作用域時(shí)不管的。而$apply還要評估父作用域,這就大大消耗了性能。
2.$timeout
用$timeout去代替你的setTimeout,$timeout作為Angularjs的自帶服務(wù),當(dāng)然時(shí)更契合Angularjs環(huán)境啦。它會(huì)隱性觸發(fā)digest循環(huán),而且它會(huì)延遲執(zhí)行,會(huì)在上一個(gè)digest循環(huán)完成后的下一刻,觸發(fā)digest循環(huán),這樣就不會(huì)出現(xiàn)上文所說的
$digest already in progress
我們把setTime的代碼放到$timeout中
$timeout(function(){ $scope.name = '李四'; },500)
這就能正常工作了,看,沒有討厭的apply了!
3.$evalAsync
最推薦的應(yīng)該時(shí)這個(gè)方法了。如果當(dāng)前正好有一個(gè)digest循環(huán)在執(zhí)行,那么它就會(huì)把導(dǎo)致digest循環(huán)的操作,放到當(dāng)前digest循環(huán)中去執(zhí)行。而$timeout是要等到當(dāng)前digest循環(huán)執(zhí)行完,再執(zhí)行一次digest循環(huán)才可以。所以evalAsync執(zhí)行更快,性能更好。我們可以像$timeout那樣去調(diào)用它,即
$scope.$evalAsync( function( $scope ) { console.log( "$evalAsync" ); } );
以上,就是今天要說的全部內(nèi)容。Angularjs中還藏著許多奧秘,和更好的使用方法,希望大家可以深入地研究,分享出更好的文章。
下面是可執(zhí)行的代碼,大家可以探究探究:https://codepen.io/hanwolfxue/pen/yEZbYQ
總結(jié)
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,如果有疑問大家可以留言交流,謝謝大家對腳本之家的支持。
相關(guān)文章
淺談angular4 ng-content 中隱藏的內(nèi)容
本篇文章主要介紹了淺談angular4 ng-content 中隱藏的內(nèi)容,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2017-08-08詳解用webpack2搭建angular2的項(xiàng)目
本篇文章主要介紹了詳解用webpack2搭建angular2的項(xiàng)目 ,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2017-06-06Angular統(tǒng)一注入器unified injector簡化依賴關(guān)系管理
這篇文章主要為大家介紹了Angular統(tǒng)一注入器unified injector簡化依賴關(guān)系管理的使用方法實(shí)例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-10-10淺析Angular 實(shí)現(xiàn)一個(gè)repeat指令的方法
這篇文章主要介紹了Angular 實(shí)現(xiàn)一個(gè)repeat指令的方法,本文給大家介紹的非常詳細(xì),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2019-07-07AngularJS 將再發(fā)布一個(gè)重要版本 然后進(jìn)入長期支持階段
目前團(tuán)隊(duì)正在開發(fā) AngularJS 1.7.0,而 1.7 的開發(fā)周期將一直持續(xù)到 2018 年 6 月 30 日2018-01-01淺談Angularjs link和compile的使用區(qū)別
下面小編就為大家?guī)硪黄獪\談Angularjs link和compile的使用區(qū)別。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2016-10-10淺談angular表單提交中ng-submit的默認(rèn)使用方法
今天小編就為大家分享一篇淺談angular表單提交中ng-submit的默認(rèn)使用方法。具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧2018-09-09Angular移動(dòng)端頁面input無法輸入的解決方法
下面小編就為大家?guī)硪黄狝ngular移動(dòng)端頁面input無法輸入的解決方法。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2017-11-11在AngularJS框架中處理數(shù)據(jù)建模的方式解析
這篇文章主要介紹了在AngularJS框架中處理數(shù)據(jù)建模的方式,作者同時(shí)也對AngularJS使用過程中的一些"坑"作了介紹,需要的朋友可以參考下2016-03-03