全面解析Angular中$Apply()及$Digest()的区别


Posted in Javascript onAugust 04, 2016

$apply()和$digest()在AngularJS中是两个核心概念,但是有时候它们又让人困惑。而为了了解AngularJS的工作方式,首先需要了解$apply()和$digest()是如何工作的。这篇文章旨在解释$apply()和$digest()是什么,以及在日常的编码中如何应用它们。

1、探索$apply()和$digest()

1.1、认识双向数据绑定和$watch();

AngularJS提供了一个非常酷的特性叫做双向数据绑定(Two-way Data Binding),这个特性大大简化了我们的代码编写方式。数据绑定意味着当View中有任何数据发生了变化,那么这个变化也会自动地反馈到scope的数据上,也即意味着scope模型会自动地更新。类似地,当scope模型发生变化时,view中的数据也会更新到最新的值。那么AngularJS是如何做到这一点的呢?当你写下表达式如{{ aModel }}时,AngularJS在幕后会为你在scope模型上设置一个watcher,它用来在数据发生变化的时候更新view。这里的watcher和你会在AngularJS中设置的watcher是一样的:

$scope.$watch(‘aModel', function(newValue, oldValue) { 
//update the DOM with newValue 
});

传入到$watch()中的第二个参数是一个回调函数,该函数在aModel的值发生变化的时候会被调用。当aModel发生变化的时候,这个回调函数会被调用来更新view这一点不难理解,但是,还存在一个很重要的问题!AngularJS是如何知道什么时候要调用这个回调函数呢?换句话说,AngularJS是如何知晓aModel发生了变化,才调用了对应的回调函数呢?它会周期性的运行一个函数来检查scope模型中的数据是否发生了变化吗?好吧,这就是$digest循环的用武之地了。

在$digest循环中,watchers会被触发。当一个watcher被触发时,AngularJS会检测scope模型,如何它发生了变化那么关联到该watcher的回调函数就会被调用。那么,下一个问题就是$digest循环是在什么时候以各种方式开始的?

在调用了$scope.$digest()后,$digest循环就开始了。假设你在一个ng-click指令对应的handler函数中更改了scope中的一条数据,此时AngularJS会自动地通过调用$digest()来触发一轮$digest循环。当$digest循环开始后,它会触发每个watcher。这些watchers会检查scope中的当前model值是否和上一次计算得到的model值不同。如果不同,那么对应的回调函数会被执行。调用该函数的结果,就是view中的表达式内容(译注:诸如{{ aModel }})会被更新。除了ng-click指令,还有一些其它的built-in指令以及服务来让你更改models(比如ng-model,$timeout等)和自动触发一次$digest循环。

目前为止还不错!但是,有一个小问题。在上面的例子中,AngularJS并不直接调用$digest(),而是调用$scope.$apply(),后者会调用$rootScope.$digest()。因此,一轮$digest循环在$rootScope开始,随后会访问到所有的children scope中的watchers。

Note: $scope.$apply()会自动地调用$rootScope.$digest()。

$apply()方法有两种形式:

第一种会接受一个function作为参数,执行该function并且触发一轮$digest循环。

第二种会不接受任何参数,只是触发一轮$digest循环。我们马上会看到为什么第一种形式更好。

1.2、什么时候手动调用$apply()方法?

如果AngularJS总是将我们的代码wrap到一个function中并传入$apply(),以此来开始一轮$digest循环,那么什么时候才需要我们手动地调用$apply()方法呢?实际上,AngularJS对此有着非常明确的要求,就是它只负责对发生于AngularJS上下文环境中的变更会做出自动地响应(即,在$apply()方法中发生的对于models的更改)。AngularJS的built-in指令就是这样做的,所以任何的model变更都会被反映到view中。但是,如果你在AngularJS上下文之外的任何地方修改了model,那么你就需要通过手动调用$apply()来通知AngularJS。这就像告诉AngularJS,你修改了一些models,希望AngularJS帮你触发watchers来做出正确的响应。

比如,如果你使用了JavaScript中的setTimeout()来更新一个scope model,那么AngularJS就没有办法知道你更改了什么。这种情况下,调用$apply()就是你的责任了,通过调用它来触发一轮$digest循环。类似地,如果你有一个指令用来设置一个DOM事件listener并且在该listener中修改了一些models,那么你也需要通过手动调用$apply()来确保变更会被正确的反映到view中。

让我们来看一个例子。加入你有一个页面,一旦该页面加载完毕了,你希望在两秒钟之后显示一条信息。你的实现可能是下面这个样子的:

html:

<body ng-app=“myApp”> 
<div ng-controller=“MessageController”> 
Delayed Message: {{message}} 
</div> 
</body>

JavaScript:

/* What happens without an $apply() */ 
angular.module(‘myApp',[]).controller(‘MessageController', function($scope) { 
$scope.getMessage = function() { 
setTimeout(function() { 
$scope.message = ‘Fetched after 3 seconds'; 
console.log(‘message:'+$scope.message); 
}, 2000); 
} 
$scope.getMessage(); 
});

通过运行这个例子,你会看到过了两秒钟之后,控制台确实会显示出已经更新的model,然而,view并没有更新。原因也许你已经知道了,就是我们忘了调用$apply()方法。因此,我们需要修改getMessage(),如下所示:

/* What happens with $apply */ 
angular.module(‘myApp',[]).controller(‘MessageController', function($scope) { 
$scope.getMessage = function() { 
setTimeout(function() { 
$scope.$apply(function() { 
//wrapped this within $apply 
$scope.message = ‘Fetched after 3 seconds'; 
console.log(‘message:' + $scope.message); 
}); 
}, 2000); 
} 
$scope.getMessage(); 
});

如果你运行了上面的例子,你会看到view在两秒钟之后也会更新。唯一的变化是我们的代码现在被wrapped到了$scope.$apply()中,它会自动触发$rootScope.$digest(),从而让watchers被触发用以更新view。

Note:顺便提一下,你应该使用$timeout service来代替setTimeout(),因为前者会帮你调用$apply(),让你不需要手动地调用它。

而且,注意在以上的代码中你也可以在修改了model之后手动调用没有参数的$apply(),就像下面这样:

$scope.getMessage = function() { 
setTimeout(function() { 
$scope.message = ‘Fetched after two seconds'; 
console.log(‘message:' + $scope.message); 
$scope.$apply(); //this triggers a $digest 
}, 2000); 
};

以上的代码使用了$apply()的第二种形式,也就是没有参数的形式。需要记住的是你总是应该使用接受一个function作为参数的$apply()方法。这是因为当你传入一个function到$apply()中的时候,这个function会被包装到一个try…catch块中,所以一旦有异常发生,该异常会被$exceptionHandler service处理。

使用 $apply()的情况如下:

•通常可以依赖于Angular提供的可用于视图中的任意指令来调用 $apply() 。所有 ng-[event]指令(比如 ng-click 、 ng-keypress )都会调用 $apply() 。

•此外还可以依赖于一系列Angular内置的服务来调用 $digest() 。比如 $http 服务会在XHR请求完成并触发更新返回值(promise)之后调用 $apply() 。

•无论何时我们手动处理事件,使用第三方框架(比如jQuery、Facebook API) ,或者调用setTimeout() ,都可以使用 $apply() 函数让Angular返回 $digest 循环。

调用setTimeout():

<!DOCTYPE html>
<html ng-app="myApp">
<head>
<title>$scope.$apply()用法</title>
<meta charset="utf-8">
<script src="http://apps.bdimg.com/libs/angular.js/1.4.6/angular.min.js"></script>
</head>
<body>
<div id="div1" ng-controller="mytext"> 
<div>{{text}}</div> 
<input id="btn" type="button" value="jquery-event"></input> 
</div> 
</body>
</html>
<script type="text/javascript">
var myModule = angular.module('myApp', []); 
myModule.controller("mytext",function($scope){ 
$scope.text = "place"; 
setTimeout(function(){ 
$scope.text = "value setted after time out"; 
$scope.$apply();//必需手动进行脏值检测,否则数据无法刷新到界面 
},1000); 
}); 
</script>

使用第三方框架(比如jQuery、Facebook API):

<!DOCTYPE html>
<html ng-app="myApp">
<head>
<title>$scope.$apply()用法</title>
<meta charset="utf-8">
<script src="https://cdn.jsdelivr.net/jquery/3.1.0/jquery.min.js"></script>
<script src="http://apps.bdimg.com/libs/angular.js/1.4.6/angular.min.js"></script>
</head>
<body>
<div id="div1" ng-controller="mytext"> 
<div>{{text}}</div> 
<input id="btn" type="button" value="jquery-event"></input> 
</div> 
</body>
</html>
<script type="text/javascript">
var myModule = angular.module('myApp', []); 
myModule.controller("mytext",function($scope){ 
$scope.text = "place"; 
}); 
$(function(){ 
$("#btn").click(function(){ 
var $scope = $("#btn").scope(); 
$scope.text = "value setted in jquery"; 
$scope.$apply(); 
}); 
}) 
</script>

1.3、$digest循环会运行多少次?

当一个$digest循环运行时,watchers会被执行来检查scope中的models是否发生了变化。如果发生了变化,那么相应的listener函数就会被执行。这涉及到一个重要的问题。如果listener函数本身会修改一个scope model呢?AngularJS会怎么处理这种情况?

答案是$digest循环不会只运行一次。在当前的一次循环结束后,它会再执行一次循环用来检查是否有models发生了变化。这就是脏检查(Dirty Checking),它用来处理在listener函数被执行时可能引起的model变化。因此,$digest循环会持续运行直到model不再发生变化,或者$digest循环的次数达到了10次。因此,尽可能地不要在listener函数中修改model。

Note: $digest循环最少也会运行两次,即使在listener函数中并没有改变任何model。正如上面讨论的那样,它会多运行一次来确保models没有变化。

结语

需要记住的最重要的是AngularJS是否能检测到你对于model的修改。如果它不能检测到,那么你就需要手动地调用$apply()。

如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对三水点靠木网站的支持!

Javascript 相关文章推荐
JS Excel读取和写入操作(模板操作)实现代码
Apr 11 Javascript
jquery 插件开发 extjs中的extend用法小结
Jan 04 Javascript
JavaScript操纵窗口的方法小结
Jun 28 Javascript
Jquery倒数计时按钮setTimeout的实例代码
Jul 04 Javascript
JS操作数据库的实例代码
Oct 17 Javascript
javascript动态创建表格及添加数据实例详解
May 13 Javascript
Angular.js与Bootstrap相结合实现表格分页代码
Apr 12 Javascript
再谈Javascript中的异步以及如何异步
Aug 19 Javascript
JavaScript中关于class的调用方法
Nov 28 Javascript
webpack4.0 入门实践教程
Oct 08 Javascript
关于vue的npm run dev和npm run build的区别介绍
Jan 14 Javascript
JavaScript Image对象实现原理实例解析
Aug 26 Javascript
window.open不被拦截的简单实现代码(推荐)
Aug 04 #Javascript
js实现多图左右切换功能
Aug 04 #Javascript
window.open打开窗口被拦截的快速解决方法
Aug 04 #Javascript
浅谈js中子页面父页面方法 变量相互调用
Aug 04 #Javascript
JS图片等比例缩放方法完整示例
Aug 03 #Javascript
JS获取url参数、主域名的方法实例分析
Aug 03 #Javascript
JavaScript中的冒泡排序法
Aug 03 #Javascript
You might like
AM/FM收音机的安装与调试
2021/03/02 无线电
php 前一天或后一天的日期
2008/06/28 PHP
php判断输入不超过mysql的varchar字段的长度范围
2011/06/24 PHP
THINKPHP支持YAML配置文件的设置方法
2015/03/17 PHP
php传值赋值和传地址赋值用法实例分析
2015/06/20 PHP
通过修改配置真正解决php文件上传大小限制问题(nginx+php)
2015/09/23 PHP
php+MySql实现登录系统与输出浏览者信息功能
2016/07/01 PHP
实例解析php的数据类型
2018/10/24 PHP
php中的依赖注入实例详解
2019/08/14 PHP
JQuery1.8 判断元素是否绑定事件的方法
2014/07/10 Javascript
jQuery实现的导航条切换可显示隐藏
2014/10/22 Javascript
原生javascript获取元素样式
2014/12/31 Javascript
jQuery实现信息提示框(带有圆角框与动画)效果
2015/08/07 Javascript
JavaScript 过滤关键字
2017/03/20 Javascript
Javascript中Promise的四种常用方法总结
2017/07/14 Javascript
jquery操作select常见方法大全【7种情况】
2019/05/28 jQuery
在react项目中使用antd的form组件,动态设置input框的值
2020/10/24 Javascript
Python操作json数据的一个简单例子
2014/04/17 Python
Python中为feedparser设置超时时间避免堵塞
2014/09/28 Python
详解Python中的正则表达式的用法
2015/04/09 Python
举例讲解Python面相对象编程中对象的属性与类的方法
2016/01/19 Python
Java Web开发过程中登陆模块的验证码的实现方式总结
2016/05/25 Python
Python使用numpy实现BP神经网络
2018/03/10 Python
Python 删除连续出现的指定字符的实例
2018/06/29 Python
PyGame贪吃蛇的实现代码示例
2018/11/21 Python
Python3.x+迅雷x 自动下载高分电影的实现方法
2020/01/12 Python
python文件排序的方法总结
2020/09/13 Python
CSS3教程(4):网页边框和网页文字阴影
2009/04/02 HTML / CSS
Html5游戏开发之乒乓Ping Pong游戏示例(三)
2013/01/21 HTML / CSS
canvas实现圆形进度条动画的示例代码
2017/12/26 HTML / CSS
有针对性的求职自荐信
2013/11/14 职场文书
2014年助理工程师工作总结
2014/11/14 职场文书
敲诈同学钱财检讨书范文
2014/11/18 职场文书
房地产项目合作意向书
2015/05/08 职场文书
2015年初一班主任工作总结
2015/05/13 职场文书
文书工作总结(范文)
2019/07/11 职场文书