AngularJS中的$watch(),$digest()和$apply()区分


Posted in Javascript onApril 04, 2016

AngularJS $scope里面的$watch(),$digest()和$apply()是AngularJS的核心函数,学习AngularJS必须理解这几个函数。

在绑定$scope中的变量到view的时候,AngularJS自动在内部创建一个"Watch"。"Watch"用于监听AngularJS scope中变量的改变。可以通过调用$scope.$watch()这个方法来创建"Watch"。

$scope.$digest()函数会循环访问所有的watches,并检测其所监听的$scope中的变量是否改变。如果变量发生改变,会调用该变量对应的监听函数。监听函数可以实现很多操作,比如让html里面的text文本显示最新的变量值。可见,$scope.$digest是可以触发数据绑定更新的。

大部分情况下,AngualrJS会自动调用$scope.$watch()和$scope.$digest()函数,但是在某些情况下,我们需要手动调用他们,因此,有必要了解他们是怎么工作的。

$scope.$apply()这个函数会先执行一些代码,之后在调用$scope.$digest()。所有的watches会被检测一次,相应的监听函数也会被执行。$scope.$apply()在AngularJS与其它javascript代码集成时是很有用的。

接下来我们具体的讲解下$watch(), $digest() 和 $apply()。

$watch()
$watch(watchExpression, listener, [objectEquality])

watchExpression:监听对象,可以是string或者function(scope){}

listener:监听对象发生改变时执行的回调函数function(newVal,oldVal,scope){}

objectEquality:是否深度监听,如果设置为true,它告诉Angular检查所监控的对象中每一个属性的变化。如果你希望监控数组的个别元素或者对象的属性而不是一个普通的值, 那么你应该使用它。(默认值:false)

$digest()
检测当前scope以及子scope中所有的watches,因为监听函数会在执行过程中修改model(scope中的变量),$digest()会一直被调用直到model没有再变。当调用超过10次时,$digest()会抛出一个异常"Maximum iteration limit exceeded',以此来防止程序进入一个死循环。

$apply()
$apply([exp])

exp:string或者function(scope){}

$apply()生命周期伪代码示意图如下

function $apply(expr) {
 try {
  return $eval(expr);
 } catch (e) {
  $exceptionHandler(e);
 } finally {
  $root.$digest();
 }
}

Example
下面我们通过一个例子来说明$watch,$digest和$apply。

<script>
var module = angular.module("myapp", []);
var myController1 = module.controller("myController", function($scope) {
  $scope.data = { time : new Date() };
  $scope.updateTime = function() {
    $scope.data.time = new Date();
  }
   
  document.getElementById("updateTimeButton")
      .addEventListener('click', function() {
    console.log("update time clicked");
    $scope.data.time = new Date();
  });
});
</script>
<body ng-app="myapp">
<div ng-controller="myController">
  {{data.time}}
 
  <br/>
  <button ng-click="updateTime()">update time - ng-click</button>
  <button id="updateTimeButton" >update time</button>
</div>
</body>

这段代码会绑定$scope.data.time到HTML中显示出来,同时这个绑定会自动创建一个watch来监听$scope.date.time的变化。此外,这里还有2个按钮,第一个按钮是通过ng-click Directive来调用$scope.updateTime方法,之后AngularJS会自动执行$scope.$digest()使最新的时间显示到HTML中。第二个按钮是通过javascript代码添加一个点击事件,以此来更新HTML中的时间。但是第二个按钮是不能工作的,它的解决办法是在点击事件的最后手动的去调用$scope.$digest()方法,如下:

document.getElementById("updateTimeButton")
    .addEventListener('click', function() {
  console.log("update time clicked");
  $scope.data.time = new Date();
  $scope.$digest();
});

另外一个解决办法是调用$scope.$apply(),如下:

document.getElementById("updateTimeButton")
    .addEventListener('click', function() {
  $scope.$apply(function(){
      console.log("update time clicked");
      $scope.data.time = new Date();
    }
  );
});

以上就是本文的全部内容,希望对大家的学习有所帮助。

Javascript 相关文章推荐
js程序中美元符号$是什么
Jun 05 Javascript
Extjs中通过Tree加载右侧TabPanel具体实现
May 05 Javascript
jQuery复制表单元素附源码分享效果演示
Sep 30 Javascript
javascript制作照片墙及制作过程中出现的问题
Apr 04 Javascript
JS实现一个简单的日历
Feb 22 Javascript
JavaScript之排序函数_动力节点Java学院整理
Jun 30 Javascript
vue学习笔记之v-if和v-show的区别
Sep 20 Javascript
jQuery简单判断值是否存在于数组中的方法示例
Apr 17 jQuery
vuejs使用axios异步访问时用get和post的实例讲解
Aug 09 Javascript
微信小程序点击滚动到指定位置的实现
May 22 Javascript
vue-video-player视频播放器使用配置详解
Oct 23 Javascript
利用javaScript处理常用事件详解
Apr 14 Javascript
Angular 根据 service 的状态更新 directive
Apr 03 #Javascript
jQuery中的Deferred和promise 的区别
Apr 03 #Javascript
再次谈论React.js实现原生js拖拽效果引起的一系列问题
Apr 03 #Javascript
jQuery qrcode生成二维码的方法
Apr 03 #Javascript
Node.js 应用跑得更快 10 个技巧
Apr 03 #Javascript
AngularJs 60分钟入门基础教程
Apr 03 #Javascript
深入浅析JSON.parse()、JSON.stringify()和eval()的作用详解
Apr 03 #Javascript
You might like
全国FM电台频率大全 - 6 辽宁省
2020/03/11 无线电
php获取url字符串截取路径的文件名和扩展名的函数
2010/01/22 PHP
php性能优化分析工具XDebug 大型网站调试工具
2011/05/22 PHP
使用PHP下载CSS文件中的图片的代码
2013/09/24 PHP
PHP中使用glob函数实现一句话删除某个目录下的所有文件
2014/07/22 PHP
thinkphp使用phpmailer发送邮件的方法
2014/11/24 PHP
WordPress中的shortcode短代码功能使用详解
2016/05/17 PHP
laravel框架中视图的基本使用方法分析
2019/11/23 PHP
PHP7 list() 函数修改
2021/03/09 PHP
TNC vs BOOM BO3 第二场2.13
2021/03/10 DOTA
让广告代码不再影响你的网页加载速度
2006/07/07 Javascript
简单的前端js+ajax 购物车框架(入门篇)
2011/10/29 Javascript
struts2+jquery+json实现异步加载数据(自写)
2013/06/24 Javascript
浅谈Javascript的静态属性和原型属性
2015/05/07 Javascript
总结JavaScript的正则与其他语言的不同之处
2016/08/25 Javascript
JavaScript提高加载和执行效率的方法
2017/02/03 Javascript
angularJS利用ng-repeat遍历二维数组的实例代码
2017/06/03 Javascript
AngularJS实现自定义指令及指令配置项的方法
2017/11/20 Javascript
vue组件name的作用小结
2018/05/23 Javascript
vue2.0 element-ui中el-select选择器无法显示选中的内容(解决方法)
2018/08/24 Javascript
JavaScript canvas绘制圆弧与圆形
2020/02/18 Javascript
[07:25]DOTA2-DPC中国联赛2月5日Recap集锦
2021/03/11 DOTA
Python Web框架Pylons中使用MongoDB的例子
2013/12/03 Python
Python矩阵常见运算操作实例总结
2017/09/29 Python
使用Python机器学习降低静态日志噪声
2018/09/29 Python
面向新手解析python Beautiful Soup基本用法
2020/07/11 Python
使用Python Tkinter实现剪刀石头布小游戏功能
2020/10/23 Python
检查接待方案
2014/02/27 职场文书
竞选副班长演讲稿
2014/04/24 职场文书
应聘英语教师求职信
2014/04/24 职场文书
党员学习群众路线教育实践活动对照检查材料
2014/09/23 职场文书
教师学习中国梦心得体会
2016/01/05 职场文书
MySQL子查询中order by不生效问题的解决方法
2021/08/02 MySQL
python实现局部图像放大
2021/11/17 Python
Java中的随机数Random
2022/03/17 Java/Android
【海涛教你打DOTA】虚空假面第一视角骨弓3房29杀
2022/04/01 DOTA