AngularJS 多指令Scope問題的解決
問題描述
不確定度指令,傳入?yún)⒘款悇e,然后該指令列出該類別下的所有不確定度。
新增頁面用到了三個該指令,只有最后一個成功,前兩個都沒有數(shù)據(jù)。
探究源碼
以下是指令源碼:
'use strict'; /** * @ngdoc directive * @name webappApp.directive:yunzhiAccuracyUncertainty * @description * # yunzhiAccuracyUncertainty * 不確定度指令 * zhangxishuo */ angular.module('webappApp') .directive('yunzhiAccuracyUncertainty', function($filter) { return { templateUrl: 'views/directive/yunzhiAccuracyUncertainty.html', restrict: 'E', scope: { parameterCategory: '=', // 參量類別 ngModel: '=' // 不確定度 }, link: function postLink(scope, element, attrs) { var self = this; // 初始化 self.init = function() { // 初始化不確定度空列表 scope.accuracyList = []; // 監(jiān)聽參量類別 scope.$watch('parameterCategory', self.watchParameterCategory); // 監(jiān)聽不確定度 scope.$watch('ngModel', self.watchNgModel); }; // 監(jiān)聽參量類別 self.watchParameterCategory = function(newValue) { if (newValue && newValue.id) { // 設(shè)置不確定度列表 scope.accuracyList = newValue.accuracyUncertaintyList; // 過濾數(shù)據(jù) self.filter(); } }; // 監(jiān)聽不確定度 self.watchNgModel = function(newValue) { if (newValue && newValue.id) { // 設(shè)置默認(rèn)選中 scope.selected = newValue; } }; // 過濾數(shù)據(jù) self.filter = function() { angular.forEach(scope.accuracyList, function(accuracy) { // 過濾不確定度 accuracy._value = $filter('yunzhiAccuracyWithUnit')(accuracy); }); }; // 更新模型 self.updateModel = function(selected) { // 更新數(shù)據(jù) scope.ngModel = selected; }; // 傳給視圖 scope.updateModel = self.updateModel; self.init(); } }; });
嘗試
嘗試打印了一下scope.accuracyList
,果然有問題。
前兩個都是空,最后一個數(shù)組有值。
想不明白,這里明明監(jiān)聽參量類別,并將scope
的accuracyList
設(shè)置了值???為什么沒有呢?
scope
嘗試打印一下scope
。
去關(guān)注scope
的$id
就行了。
依次打印的是:
504 508 // 第一個指令 506 508 // 第二個指令 508 508 // 第三個指令
前兩個指令執(zhí)行時賦值的是一個scope
,而過濾的又是另一個scope
,所以過濾不出數(shù)據(jù),最后一個是同一scope
,所以正常輸出。
原因
官方文檔
HTML Compiler
允許開發(fā)者教會瀏覽器一些新的語法,AngularJS
稱這個為指令。
Compiler
是一個遍歷DOM
去搜尋屬性的AngularJS
服務(wù),編譯分為以下兩個階段。
Compile
:遍歷DOM
并收集所有的指令,返回結(jié)果是一個linking
函數(shù)。Link
:使用scope
整合指令并產(chǎn)生動態(tài)視圖,任何scope
模型上的改變都會反映到視圖上,任何視圖上的用戶交互也會反映到scope
模型上。
指令如何編譯
AngularJS
操作DOM
節(jié)點(diǎn)而不是字符串,這很重要。但通常,你不需要關(guān)注這個,因?yàn)楫?dāng)頁面加載時,瀏覽器會自動把HTML
轉(zhuǎn)換為DOM
。
指令編譯有以下三階段:
$compile
遍歷DOM
并匹配指令,如果compiler
發(fā)現(xiàn)有匹配指令的元素,就會將該指令添加到指令列表中。一個元素可能匹配多個指令。- 一旦所有匹配
DOM
元素的指令都被確定,然后compiler
會根據(jù)優(yōu)先級對指令進(jìn)行排序。每一個指令的compile
函數(shù)都會被執(zhí)行,每一個compile
函數(shù)都有操作DOM
的機(jī)會。compile
會返回link
函數(shù),這些函數(shù)被組合成一個“組合的”link
函數(shù),它能調(diào)用每個指令返回的link
函數(shù)。 $compile
會調(diào)用上一步中的“組合的”link
函數(shù)來鏈接scope
和模板。
下面是官方的示意代碼:
// HTML字符串 var html = '<div ng-bind="exp"></div>'; // 將HTML字符串轉(zhuǎn)換為DOM模板 var template = angular.element(html); // 編譯DOM模板返回link函數(shù) var linkFn = $compile(template); // 將編譯后的模板與scope鏈接 var element = linkFn(scope); // 添加到DOM中 parent.appendChild(element);
分析
compile
只在編譯時執(zhí)行一次,只要頁面中存在一個該指令,該指令的link
方法就執(zhí)行一次。
所以,AngularJS
使用$compile
編譯我的指令,然后看我頁面中用到了三個該指令,并且都是獨(dú)立scope
,所以就創(chuàng)建了三個scope
。
然后使用這三個scope
去調(diào)用link
函數(shù)。
前面已經(jīng)提到,AngularJS
會將link
函數(shù)統(tǒng)一組合成一個“組合的”link
函數(shù),所以我們可以猜想,組合函數(shù)中的link
函數(shù)的數(shù)量與指令的數(shù)量一致,所以三次調(diào)用的是一個link
函數(shù),link
函數(shù)只有一個實(shí)例!
linkFn(scope)
將scope
傳進(jìn)去作為link
函數(shù)的入?yún)ⅰ?/p>
上面的事件監(jiān)聽都是沒毛病的,將傳入的scope
綁定到視圖,然后添加到DOM
中,然后就與這個link
函數(shù)無關(guān)了。
但是這個filter
就不行了。
第一個scope
調(diào)用,filter
功能是過濾第一個scope
的accuracyList
,第二個scope
調(diào)用,filter
功能是過濾第二個scope
的accuracyList
。
所以第三次執(zhí)行時,第三個scope
將之前的兩個都覆蓋了,link
函數(shù)中的filter
的作用變成了過濾最后一個scope
的accuracyList
。
<!-- 不確定度 --> <ui-select ng-model="selected" theme="bootstrap" ng-change="updateModel(selected)"> <ui-select-match placeholder="請選擇"> {{ $select.selected._value }} </ui-select-match> <ui-select-choices repeat="accuracy in accuracyList"> <div ng-bind-html="accuracy._value"></div> </ui-select-choices> </ui-select>
所以這里下拉框顯示的是不確定度過濾后的_value
的值,這里的空字符串看起來不明顯,加上test
測試一下。
所以,這塊視圖綁定的scope
是正確的,只是時間監(jiān)聽之后去過濾數(shù)據(jù),因?yàn)檫^濾的并不是當(dāng)前scope
的數(shù)據(jù),所以accuracy._value
就沒有值,是undefined
,所以顯示一個空的字符串。
解決方案
明白了原理之后解決問題自然易如反掌,只需將filter
與scope
獨(dú)立即可,這樣就不受每次執(zhí)行不同scope
的影響了。
總結(jié)
很多東西,書上是沒有的,需要我們自己去發(fā)現(xiàn),去分析,去解決。
翻開了之前遇到指令編譯問題時從別人博客里學(xué)習(xí)來的手動編譯方法。
angular.module('webappApp') .directive('reCompile', function($compile) { return { restrict: 'A', link: function postLink(scope, element, attrs) { // 監(jiān)聽使用該指令的元素上的ngBindHtml attrs.$observe('ngBindHtml', function() { // 如果元素使用了ngBindHtml指令 if (attrs.ngBindHtml) { // 重新編譯 $compile(element[0].children)(scope); } }); } }; });
記得之前的需求是,數(shù)據(jù)經(jīng)過過濾器過濾,返回的是一段HTML
代碼,雖然使用ng-bind-html
能將該段代碼添加到DOM
中,但是這段代碼中有指令,因?yàn)樵撝噶畈皇浅跏紩r就有的,所以,這個指令是不會被編譯的。
所以需要編寫一個重新編譯的指令,手動編譯動態(tài)創(chuàng)建的指令。
記得當(dāng)時,看這段代碼也不是那么完全理解,現(xiàn)在學(xué)習(xí)完指令的編譯之后,再去翻看之前的代碼,一切原來是如此簡單。
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
基于Angularjs+mybatis實(shí)現(xiàn)二級評論系統(tǒng)(仿簡書)
這篇文章主要為大家詳細(xì)介紹了基于Angularjs+mybatis實(shí)現(xiàn)二級評論系統(tǒng),模仿簡書效果制作,具有一定的參考價值,感興趣的小伙伴們可以參考一下2017-02-02Angular.js實(shí)現(xiàn)掃碼槍掃碼并生成二維碼
這篇文章主要為大家介紹了Angular.js實(shí)現(xiàn)掃碼槍掃碼并生成二維碼示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-03-03Angular實(shí)現(xiàn)跨域(搜索框的下拉列表)
angular.js 自帶jsonp,實(shí)現(xiàn)跨域,下面來實(shí)現(xiàn)搜索框的下拉列表功能,本文思路明確,非常不錯,具有參考借鑒價值,需要的朋友參考下吧2017-02-02AngularJS 將再發(fā)布一個重要版本 然后進(jìn)入長期支持階段
目前團(tuán)隊正在開發(fā) AngularJS 1.7.0,而 1.7 的開發(fā)周期將一直持續(xù)到 2018 年 6 月 30 日2018-01-01Angular中自定義Debounce Click指令防止重復(fù)點(diǎn)擊
本篇文章主要介紹了Angular中自定義Debounce Click指令詳解,具有一定的參考價值,感興趣的小伙伴們可以參考一下2017-07-07AngularJS基礎(chǔ) ng-selected 指令簡單示例
本文主要介紹AngularJS ng-selected 指令,這里對ng-selected 指令的基礎(chǔ)資料做了詳細(xì)介紹,并附有示例代碼,有需要的小伙伴可以參考下2016-08-08angularjs中控制視圖的控制器的兩種注入依賴項及服務(wù)的寫法小結(jié)
在AngularJS中,控制器的依賴注入有兩種方法:顯式依賴注入和隱匿依賴注入,顯式依賴注入通過使用字符串?dāng)?shù)組形式來注入依賴項,本文給大家介紹angularjs中控制視圖的控制器的兩種注入依賴項及服務(wù)的寫法,感興趣的朋友一起看看吧2024-09-09angular實(shí)現(xiàn)頁面打印局部功能的思考與方法
這篇文章主要給大家介紹了關(guān)于angular實(shí)現(xiàn)頁面打印局部功能的思考與方法,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧。2018-04-04