本文首发于华为云社区
原文连接:https://bbs.huaweicloud.com/blogs/e19336d0889f11e89fc57ca23e93a89fjavascript
聊聊 Angularjs1.x中那些活见鬼的事情。html
html-Controller的双向数据绑定,在开发中很是常见,也是Angularjs1.x的宣传点之一,使用中并无太多问题。java
也就是从视图层流向模型层,原生html中须要使用表单元素(例如input
标签)来收集用户输入信息,Angularjs中经过在表单元素上使用ng-model
标签,当用户输入信息时,同步将用户输入的信息赋值给controller中的变量:安全
<body ng-app="myApp"> <div id="main" ng-controller="myCtrl"> <p>改变输出值:</p> <input type="text" ng-model="testInfo.content" ng-change="showInput()"> </div> <script src="./angular.min.js"></script> <script> angular.module('myApp',[]) .controller('myCtrl',['$scope',function($scope){ $scope.showInput = function() { console.log($scope.testInfo.content); } }]); </script> </body>
在页面上输入1234567便可看到,每次在页面输入数字后,控制台输出的$scope,testInfo.content
的值都和页面保持一致:app
也就是从模型层流向数据层,当controller中的数据模型变量发生变化后,Angularjs又会根据数据模型的值去改变ng-model
指令绑定的表单元素的值,使用ng-bind
指令也能够被动得到来自controller的数据流。ide
咱们编写以下demo进行测试:函数
<body ng-app="myApp"> <div id="main" ng-controller="myCtrl"> <button ng-click="add()">+1</button> <p>改变输出值:</p> <input type="text" ng-model="testInfo.content"> <p>使用ng-bind绑定的标签:</p> <p ng-bind="testInfo.content"></p> </div> <script src="./angular.min.js"></script> <script> angular.module('myApp', []) .controller('myCtrl', ['$scope', function($scope) { //初始化 $scope.testInfo = { content: 0 } $scope.add = function () { $scope.testInfo.content += 1; console.log($scope.testInfo.content); } }]); </script> </body>
demo中,每次点击+1按钮,$scope.testInfo.content
的值会增长1,咱们能够看到页面上的结果:测试
来看看第一个活见鬼的例子,demo跟上面很相似,只是将鼠标点击触发的方式改为了定时器自动触发:设计
<body ng-app="myApp"> <div id="main" ng-controller="myCtrl"> <button ng-click="add()">+1</button> <p>改变输出值:</p> <input type="text" ng-model="testInfo.content"> <p>使用ng-bind绑定的标签:</p> <p ng-bind="testInfo.content"></p> </div> <script src="./angular.min.js"></script> <script> angular.module('myApp', []) .controller('myCtrl', ['$scope', function($scope) { //初始化 $scope.testInfo = { content: 0 } //定时自增 setInterval(function () { $scope.testInfo.content += 1; console.log('$scope.testInfo.content的值如今是:',$scope.testInfo.content); },1000) }]); </script> </body>
你会活见鬼地发现,数据模型一直在变,可是页面却没有刷新:双向绑定
这里就是
Angularjs1.X
双向数据绑定中的第一个坑 ,你会发现$scope上绑定的数据模型和html中显示的内容有时候并非实时关联的。这其实和Angularjs1.X
的执行机制有关系。
若是咱们本身来考虑,javascript中有一个变量的值发生了变化,如今要将这个值同步到html页面上,须要怎么作呢?咱们须要获取到这个DOM元素,而后改变它的innerHTML
属性,若是是表单元素就修改value
。其实Angularjs
也是这样作的,只不过使用了本身的封装的方法——$apply()。那么此处的问题其实就在于,在setInterval
的回调函数中去修改数据模型的值时,没有触发$apply()方法来更新视图,而经过调用Angularjs
封装的ng-*
方法(例如ng-click
点击方法)来修改视图模型时,会自动触发$apply()方法,视图也就同步刷新了。
解决方案1
使用Angularjs
封装过的$interval
服务来实现定时任务,感兴趣的读者能够本身看一下Angularjs
源码中$intervalProvider
的部分,就会发如今方法最后的地方调用了$rootScope.$apply()
。
解决方案2
若是依然使用javascript原生的定时方法,那么则须要在修改完视图的数据模型后,手动调用$scope.$apply()
方法来将数据模型的变更同步到html页面中。
除了controller与html中的双向绑定,Angularjs
中还有另外一个双向数据绑定,那就是controller与directive之间的绑定。绑定的形式有不少种,咱们先来看一下最多见的双向绑定。
在设定自定义指令的scope
参数时,将属性的值设置为=
就能够实现双向数据绑定,这里API的解释是:
父级controller中的指定变量会与自定义指令link函数中的变量相互影响。
下面的实例中,咱们将看看controller中的数据模型$scope.testInfo.content
的值与自定义指令中scope.pagination
如何相互影响,是否如定义所说这里的绑定真的是双向的。示例界面以下(demo源码请见附件demo.html文件):
+1
按钮,Scope.testInfo.content
的值都会增长1show $scope.testInfo
,控制台都会打印出$scope.testInfo
的值scope.pagination
的值,并将该值进行自增接下来的测试操做,咱们将按照以下的流程进行:
+1
按钮,再点击5次数字标签show $scope.testInfo
按钮随着上一节的操做步骤,咱们一块儿来见证双向数据绑定中又一次闹鬼事件:
点击5次+1
按钮,再点击5次数字标签
结果为:
咱们看到,第一次点击数字标签时,控制台打出了link函数中scope.pagination
的值为5,这说明$scope.testInfo.content
的值被传递给了自定义指令中的scope.pagination
,也就是说数据从controller流向了directive。而当咱们再点击4次数字标签(一共点了5次)后,从控制台能够看出,scope.pagination
的值已经成为10,而页面上使用ng-bind
指令获取到的结果却依旧是5。也就是说,数据从没有从directive流向controller。是否是有一种被骗的感受?别着急,接着看。
点击show $scope.testInfo
按钮
结果为:
当咱们点击show $scope.testInfo
时,控制台打印出了$scope.testInfo.content的值为5,这下证据坐实了,明明说好的双向数据绑定,然而当自定义指令中的scope.pagination
改变时,$scope.testInfo.content
并无跟着一块儿改变。But!!!!咱们会发现,这个show $scope.testInfo
点下去之后,页面上经过ng-bind
绑定的值却变成了10。也就是说,数据又从directive流回了controller。
官方建议使用$watch
方法来追踪scope中的变量,而当咱们这样作时,会发现$watch
函数仅能追踪到那些经过修改controller中的数据模型而影响link函数中变量的行为并更新视图。
这里就是
Angularjs1.X
双向数据绑定中的第二个坑,controller和directive中所谓的双向数据绑定,并不能追踪指定变量的全部变化,并且不是同步完成的。
其实这里的问题仍然和Angularjs
的运行机制有关,解决方案以下:
解决方案1
使用自定义指令的templateUrl
属性替换当前指令的模板,使用ng-click
指令来绑定一个点击响应函数,在响应函数中改变scope.piganation
的值。
解决方案2
在手动绑定的监听回调中,修改自定义指令做用域内的变量后,使用scope.$emit( )
方法通知其父级controller,并在controller中使用$scope.$on( )
方法监听同名事件,并修改对应的数据模型的值。
解决方案3
每当改变自定义指令中的变量值后,调用scope.$apply()
方法,将directive中的变量值同步至controller的数据模型以及页面。
Angularjs中的双向数据绑定,是经过一种叫作*"脏循环检查(dirty-checking)"的机制实现的。
其基本过程是这样的,每当咱们使用ng-model
或ng-bind
指令将数据模型中的某个变量值和html页面上某个标签的内容联系起来时,Angular就会把这些变量放进一个WatchCollection
的集合中,并自动帮咱们来监控这些变量。每当WatchCollection
中有变量出现变更时,Angular就会遍历WatchCollection
来查看是否有其余监控中的变量也被影响,每当有一个变量被影响,Angular都会在遍历后再进行一次遍历,直到某一次遍历后WatchCollection
中的变量都没有变化,则Angular会认为当前的改动已经稳定了,而后才会将数据模型的变化同步到DOM元素上去,也就实现了数据绑定。
咱们能够把WatchCollection
理解为当前页面的一种抽象,其中包含着页面上全部有可能发生变化的部分。
想要在Angularjs
项目中更加稳定地使用双向数据绑定,笔者的建议是:
在
Angularjs
项目中,尽量地使用Angular告诉你的方式去编写所但愿实现的功能。
咱们能够回顾一下上面在使用双向数据绑定发生异常时的场景:
$interval
,$timeout
服务)ng-click
来实现点击事件的监听)你会发现,每当本身没有按照Angular的方式去编写代码,或者没有按照一个模块设计的初衷去使用它时,就没法确切地获得指望的结果。这是很容易理解的,若是你没有按照Angular要求的方式书写代码,凭什么指望它对你的代码作出100%正确的回应呢?至于上述两种数据绑定中出现问题的解决方案,上文已经有所说起,此处再也不赘述。
许多人都据说过"尽可能不要在controller中操做DOM"这句话,实际上它并不意味着你在controller中操做DOM会致使程序报错,而是在说若是你同时使用jQuery
和Angular
两套系统来管理本身的代码,但又没有按照官方指定的方式来规避它们之间的冲突,那代码极可能会变得不稳定。想一想当年腾讯电脑管家和360安全卫士将你的电脑卡死的场景,你就明白这样作的结果了。
笔者曾经看过这样一段话,以为深有感触:
所谓高手,是指那些
熟知套路
且创意无穷
的人。而高手之间的较量,归根结底都是基本功
的比拼。
愿有朝一日,你也能成为高手。
本文首发于华为云社区
原文连接:https://bbs.huaweicloud.com/blogs/e19336d0889f11e89fc57ca23e93a89f