Angular:手动脏检查/$apply/$digest和监控对象/$watch
声明:借鉴好多chm资料、视频、PDF总结如下:
一、$apply的引入
View
<div ng-app="">
<div ng-controller="firstController">
<input ng-model="date"/>
{{date}}
</div>
</div>
原生js函数,不能实现刷新(因为没有实现双向绑定(脏检查))
var firstController = function($scope) {
$scope.date = new Date().getSeconds();
setInterval(function() {
//并没有触发 脏检查
$scope.date = new Date().getSeconds();
}, 1000)
}
原生js函数,在变量改变外面加上$scope.apply(fun),手动实现脏检查,实现刷新
var firstController = function($scope) {
$scope.date = new Date().getSeconds();
setInterval(function() {
$scope.$apply(function() {
$scope.date = new Date().getSeconds();
//....会去触发脏检查
})
}, 1000)
}
Angular内置函数,默认已经实现脏检查,实现刷新
var firstController = function($scope, $interval) {
$scope.date = new Date().getSeconds();
$interval(function() {
$scope.date = new Date().getSeconds();
}, 1000)
}
原生js:setInterval(fun)<==>$interval(fun) 一直循环执行
原生js:setTimeout(fun)<==>$TimeOut(fun) 执行一次
gif图结果


解释
以`ng-开头的` `事件` `指令`和默认`内置的函数`最后`都会执行脏检查`。
`原声js`要实现如此功能,必须`在变量改变外面加上$scope.apply(fun)`,这样会将这个`apply内的变量复制出一份`,`新值和旧值(备份)对比`,实现双向绑定
二、$watch监听一个对象
$watch(watchExpression, [listener], [objectEquality]);
两个参数
<div ng-app="">
<div ng-controller="firstController">
<input type="text" value="" ng-model="name"/><br/>
注意:页面加载的时候执行一次,默认1:改变次数:{{count}}
<br/>
注意:当次数超过5,页面改变一次,count会+2,触发2次
</div>
</div>
var firstController = function($scope) {
$scope.name = '李可';
$scope.count = 0;
// 监听一个model 当一个model每次改变时 都会触发第2个函数
$scope.$watch('name', function(newValue, oldValue) {
$scope.count=$scope.count+1;//页面加载的时候执行一次
if ($scope.count > 5) {//当次数超过5,页面改变一次,count会+2,触发2次
$scope.name = '已经大于5次了';
}
});
}
gif图结果

问题:为什么触发2次呢?
估计要分析脏检查的源码了:原谅我处于菜鸟阶段
问了一些人

三个参数 第三个为true,表示对象内部的属性。
深深地监视
<div ng-app="">
<div ng-controller="firstController">
<input type="text" value="" ng-model="data.name"/>
<input type="text" value="" ng-model="data.age"/>
<input type="text" value="" ng-model="count2"/>
</div>
</div>
var firstController = function($scope) {
$scope.data = {
name: '李可',
age: 18
}
$scope.count2 = 0;
$scope.$watch('data', function() {
++$scope.count2;
if ($scope.count2 > 3) {
$scope.data.age = '换超过3了';
}
}, true)
}
++i先加后用(同一个表达式内用)
i++先用(同一个表达式内用)后加
var a=2;++a;var b=a*3;alert(b)//9 ++a;var b=a*3;是两表达式
var c=2;c++;var d=c*3;alert(d)//9 c++;var d=c*3;是两表达式
var e=2;f=e++;alert(f)//2 f=e++ 有++表达式和=表达式 相当f=e; e=e+1
var g=2;h=++g;alert(h)//3 h=++g; 有++表达式和=表达式 相当g=g+1; h=g;
var i=2;x=++i +1;alert(x)//4 x=++i +1; 有++表达式和+表达式 相当i=i+1; x=i;
var m=2;y=m++ +1;alert(y)//3 x=i++ +1; 有++表达式和+表达式 相当y=m; m=m+1
gif图结果

注意:页面加载1次,`加1`
到临界条件时,`加2`,`以后也加1`
延伸
有了上述方法:我们可以监控 对象、数组。为了少第三个参数,具体数组操作,angular又增加
$watchCollection(obj, listener);,$watchGroup(watchExpressions, listener);
Angular:手动脏检查/$apply/$digest和监控对象/$watch的更多相关文章
- Angular DirtyChecking(脏值检查) $watch, $apply, $digest
Dirty Checking (脏值检查) Digest cycle and $scope Digest cycle and $scope First and foremost, AngularJS ...
- 理解Angular中的$apply()以及$digest()
$apply()和$digest()在AngularJS中是两个核心概念,但是有时候它们又让人困惑.而为了了解AngularJS的工作方式,首先需要了解$apply()和$digest()是如何工作的 ...
- 深入理解Angular中的$Apply()以及$Digest()
$apply()和$digest()在AngularJS中是两个核心概念,但是有时候它们又让人困惑.而为了了解AngularJS的工作方式,首先需要了解$apply()和$digest()是如何工作的 ...
- --@angularjs--理解Angular中的$apply()以及$digest()
$apply() 和 $digest() 在 AngularJS 中是两个核心概念,但是有时候它们又让人困惑.而为了了解 AngularJS 的工作方式,首先需要了解 $apply() 和 $dige ...
- (网页)理解Angular中的$apply()以及$digest()
转自CSDN: 工作有问题上CSDN上转转. $apply()和$digest()在AngularJS中是两个核心概念,但是有时候它们又让人困惑.而为了了解AngularJS的工作方式,首先需要了解$ ...
- (转) 理解Angular中的$apply()以及$digest()
原文地址:http://blog.csdn.net/dm_vincent/article/details/38705099 $apply()和$digest()在AngularJS中是两个核心概念,但 ...
- angularjs脏检查
angularjs实现了双向绑定,与vue的defineProperty不同,它的原理在于它的脏检查机制,以下做了一些总结: angular.js介绍 AngularJs是mvvm框架,它的组件是vm ...
- Angular 学习笔记 :初识 $digest , $watch , $apply,浅析用法 。
传统的浏览器事件循环 :浏览器本身一直在等待事件,并作出响应.如果你点击一个button或者在input 中输入字符,我们在 JS 中 监听这些事件并设定了回调函数,那么这些事件被触发以后,回调函数就 ...
- Angular 手动解析表达式
<!DOCTYPE html> <html> <head> <meta charset="utf-8"> <title> ...
随机推荐
- ubuntu 14.04安装mysql server & mysql client
$ sudo apt-get install mysql-server
- Ubuntu中root用户和user用户的相互切换
转:Ubuntu是最近很流行的一款Linux系统,因为Ubuntu默认是不启动root用户,现在介绍如何进入root的方法. (1)从user用户切换到root用户 不管是用图形模式登录Ubuntu, ...
- Redis/SSDB+Twemproxy的配置与使用(Mac/Linux平台)
对于redis而已,相信不少的后台开发人员一直都在使用,相比memcache而已,redis不仅可以作为key-value缓存使用,而且提供了丰富的数据结构如set.list.map等,能够实现很多复 ...
- Sql server之路 (二)登录本地服务器
安装环境 Microsoft SQL Server Management Studio Express http://www.microsoft.com/zh-cn/download/details ...
- ML 03、机器学习的三要素
机器学习算法原理.实现与实践——机器学习的三要素 1 模型 在监督学习中,模型就是所要学习的条件概率分布或决策函数.模型的假设空间包含所有可能的条件概率分布或决策函数.例如,假设决策函数是输入变量的线 ...
- C#学习笔记(七)——定义类
一.C#中类的定义 1.类的定义 class Myclass { } 默认情况下是类是内部的,即外部的项目是不可以访问这个类,相当于加internal来修饰.但实际上是不需要加入的. 如果要指定类是公 ...
- ASP.NET获取客户端IP/用户名等信息
1. 在ASP.NET中专用属性: 获取服务器电脑名:Page.Server.ManchineName 获取用户信息:Page.User 获取客户端电脑名:Page.Request.UserHostN ...
- 【spring 区别】ClassXmlAplicationContext和FileSystemXmlApplicationContext的区别
今天写一个简单的spring使用例子,遇到这个问题: 项目结构如下: 代码如下: package com.it.sxd; import java.nio.file.FileSystem; import ...
- Android 贝塞尔曲线 折线图
1.贝塞尔曲线:http://baike.baidu.com/view/60154.htm,在这里理解什么是贝塞尔曲线 2.直接上图: 3.100多行代码就可以画出贝塞尔曲线,直接上代码 packag ...
- HUE的时区问题
转自 http://molisa.iteye.com/blog/1953390 我主要是根据这个说明调整的HUE的时区问题 在使用Cloudera Hue时遇到一问题: 1. 使用Sqoop导入功 ...