关于 AngularJS 的数据绑定
单向绑定(ng-bind) 和 双向绑定(ng-model) 的区别
ng-bind
单向数据绑定($scope -> view
),用于数据显示,简写形式是 {{}}
。
1 |
<span ng-bind="val"></span> |
两者的区别在于页面没有加载完毕 {{val}}
会直接显示到页面,直到 Angular 渲染该绑定数据(这种行为有可能将 {{val}}
让用户看到);而 ng-bind
则是在 Angular 渲染完毕后将数据显示。
ng-model
是双向数据绑定($scope -> view
and view -> $scope
),用于绑定值会变化的表单元素等。
1 |
<input type="text" ng-model="val" /> |
双向数据绑定的原理
双向数据绑定意味着当 view
中有任何数据发生变化会自动地反馈到 scope
的数据上,当 scope
模型发生变化时,view
中的数据也会更新到最新的值。很显然,这需要一个监控。
事实上,AngularJS 确实在幕后为 scope
模型上设置了一个 监听队列,用来监听数据变化并更新view
。
每次绑定一个东西到 view
上时 AngularJS 就会往 $watch
队列里插入一条 $watch
,用来检测它监视的 model
里是否有变化的东西。
当浏览器接收到可以被 angular context
处理的事件时,$digest
循环就会触发。$digest
会遍历所有的 $watch
。
一次更新的操作(至少触发两次 $digest() 循环)
比如进行一次 click
操作:
1 |
<button ng-click="val=val+1">increase 1</button> |
- 按下按钮
- 浏览器接收到一个事件,进入
angular context
$digest
循环开始执行,查询每个$watch
是否变化- 由于监视
$scope.val
的$watch
报告了变化,它会强制再执行一次$digest
循环。 - 新的
$digest
循环没有检测到变化。 - 浏览器拿回控制权,更新与
$scope.val
新值相应部分的DOM
。
$digest 循环会运行多少次?
$digest
循环不会只运行一次。在当前的一次循环结束后,它会再执行一次循环用来检查是否有models
发生了变化。
这就是脏检查(Dirty Checking
),它用来处理在 listener
函数被执行时可能引起的 model
变化。因此 $digest
循环会持续运行直到 model
不再发生变化,或者 $digest
循环的次数达到了 10
次(超过 10
次后抛出一个异常,防止无限循环)。
当 $digest
循环结束时,DOM
相应地变化。
$apply() 和 $digest() 的区别
$apply
是 $scope
(或者是 direcvie
里的 link
函数中的 scope
)的一个函数,调用它会强制一次 $digest
循环(除非当前正在执行循环,这种情况下会抛出一个异常,这是我们不需要在那里执行 $apply
的标志)。
$apply()
和 $digest()
有两个区别。
1) 最直接的差异是, $apply
可以带参数,它可以接受一个函数,然后在应用数据之后,调用这个函数。所以,一般在集成非 Angular 框架(比如jQuery)的代码时,可以把代码写在这个里面调用。
2) 当调用 $digest
的时候,只触发当前作用域和它的子作用域上的监控,但是当调用 $apply
的时候,会触发作用域树上的所有监控。
什么时候手动调用 $apply() 方法?
取决于是否在 Angular 上下文环境(angular context
)。
典型的需要调用 $apply()
方法的场景是:
1) 使用了 JavaScript 中的 setTimeout()
来更新一个 scope model
2) 用指令设置一个 DOM
事件 listener
并且在该 listener
中修改了一些 models
场景一
1 |
$scope.setMsg = function() { |
运行这个例子,会看到过了两秒钟之后,控制台确实会显示出已经更新的 model
,然而,view
并没有更新。
在 $scope.getMessage
加入 $apply()
方法。
1 |
$scope.getMessage = function() { |
再运行就 OK 了。
不过,在 AngularJS 中应该尽量使用 $timeout
Service 来代替 setTimeout()
,因为前者会帮你调用 $apply()
,让你不需要手动地调用它。
1 |
$timeout(function(){ |
场景二
实现一个 click
的指令,类似以下功能:
1 |
<button ng-click="val=val+1">increase 1</button> |
directive
的编写如下:
1 |
app.directive("inc", function() { |
跟场景一的结果一样,这个时候,点击按钮,界面上的数字并不会增加。但查看调试器,发现数据确实已经增加了。
在 scope.val++;
一行后面添加 scope.$apply();
或者 scope.$digest();
就 OK 了。
$apply() 方法的两种形式
1) 无参
1 |
$scope.$apply() |
2) 有参
1 |
$scope.$apply(function(){ |
应该总使用接受一个 function
作为参数的 $apply()
方法。这是因为当传入一个 function
到$apply()
中的时候,这个 function
会被包装到一个 try…catch
块中,所以一旦有异常发生,该异常会被 $exceptionHandler service
处理。
想象一下如果有个 alert
框显示错误给用户,然后有个第三方的库进行一个网络调用然后失败了,如果不把它封装进 $apply
里面,Angular 永远不会知道失败了,alert
框就永远不会弹出来了。
在 AngularJS 中使用 $watch
常用的使用方式:
1 |
$scope.name = 'htf'; |
传入到 $watch()
中的第二个参数是一个回调函数,该函数在 name
的值发生变化的时候会被调用。
如果要监听的是一个对象,那还需要第三个参数:
1 |
$scope.data.name = 'htf'; |
表示比较的是对象的值而不是引用,如果不加第三个参数 true
,在 data.name
变化时,不会触发相应操作,因为引用的是同一引用。
参考
- 理解$watch ,$apply 和 $digest —- 理解数据绑定过程
- 理解Angular中的$apply()以及$digest()
- Angular沉思录(一)数据绑定
- 构建自己的AngularJS,第一部分:Scope和Digest
关于 AngularJS 的数据绑定的更多相关文章
- 10分钟学会AngularJS的数据绑定
前言:为什么要用AngularJS? 相信用过.NetMVC的人都知道用rezor绑定数据是一件很爽的事情,C#代码直接在前台页面中输出.然后这种比较适用于同步请求. 当我们的项目离不开异步请 ...
- AngularJS双向数据绑定
<!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title> ...
- 02、AngularJs的数据绑定
我们知道,AngularJs中的数据绑定是双向绑定的,View的改变,会改变Model,Model的改变也会改变View中的值,废话不多说,我们直接上代码. <!DOCTYPE html> ...
- AngularJS笔记---数据绑定
一.数据绑定 1.简单绑定 下面实现了一个简单的加法运算的绑定, A.ng-app:表示该div以内都在AngularJS的应用, 去掉ng-app="" 那么后面的绑定都将无效 ...
- AngularJS 的数据绑定
单向绑定(ng-bind) 和 双向绑定(ng-model) 的区别 ng-bind 单向数据绑定($scope -> view),用于数据显示,简写形式是 {{}}. 1 <span n ...
- Vue、AngularJS 双向数据绑定解剖
数据与视图的绑定与同步,最终体现在对数据的读写处理过程中,也就是 Object.defineProperty() 定义的数据 set.get 函数中.Vue 中对于的函数为 defineReactiv ...
- 关于AngularJs,数据绑定与自定义验证
最近开始着手学起了Angular,抱着好奇的心情开始研究了起来.忽然发现angular可以巧妙而方便的进行数据的绑定验证啊什么的.(当然,我只是刚开始学,所有可能有更强大的功能,只是我还没有看到) 那 ...
- AngularJS入门心得2——何为双向数据绑定
前言:谁说Test工作比较轻松,最近在熟悉几个case,差点没疯.最近又是断断续续的看我的AngularJS,总觉得自己还是没有入门,可能是自己欠前端的东西太多了,看不了几行代码就有几个常用函数不熟悉 ...
- AngularJS语法基础及数据绑定——详解各种数据绑定指令、属性应用
AngularJS简单易学,但是功能强大.特别是在构建单页面应用方面效果显著.而 数据绑定 可以说是他被广泛使用的最主要的优点.他舍弃了对DOM的操作方式,一切都由AngularJS来自动更新视图,我 ...
随机推荐
- Turn any Linux computer into SOCKS5 proxy in one command
src: http://www.catonmat.net/blog/linux-socks5-proxy/ I thought I'd do a shorter article on catonmat ...
- 第十二章 Openwrt无法识别2.0 USB 盘
今天新的U盘不能识别,还以为是U盘本身的问题,原来是缺少 kmod-usb-storage kmod-usb-storage-extras 这两个软件包. 软件包安装后重启便可识别.
- [Android 新特性] 15项大改进 Android 4.4新特性解析
腾讯数码讯(编译:刘沙) 终于,Android系统迎来了久违的重大更新——Android 4.4 KitKat,并与新旗舰Nexus 5同时问世.那么,新的系统究竟都有怎样的改进.是否值得升级呢,下面 ...
- linux系统的磁盘挂载
1.查看数据盘在没有分区和格式化数据盘之前,使用 “df –h”命令,是无法看到数据盘的,可以使用“fdisk -l”命令查看.如下图:2. 对数据盘进行分区执行“fdisk /dev/xvdb”命令 ...
- go语言基础之goto的用法
1.goto的用法 示例: package main //必须有一个main包 import "fmt" func main() { //break //break is not ...
- java学习笔记1--开发环境平台总结
本文地址:http://www.cnblogs.com/archimedes/p/java-study-note1.html,转载请注明源地址. 1.Java程序设计语言 Java程序设计语言是一种高 ...
- JDK 1.7版本的 新特性
摘自: http://yanguz123.iteye.com/blog/1934766 Jdk1.7的新特性: 1,switch中可以使用字串 Java代码: String s = "tes ...
- 用mysqlslap对MySQL进行压力测试
MySQL5.1地的确提供了好多有力的工具来帮助我们DBA进行数据库管理.现在看一下这个压力测试工具mysqlslap.关于他的选项手册上以及--help介绍的很详细.我解释一下一些常用的选项.这里要 ...
- Mysql触发器、模糊查找、存储过程、内置函数
原本觉得Mysql的一些知识还是差不多了,但是在实际上在项目上用的时候,发现什么都忘记了.现在重新回顾一下,顺便做个笔记. 触发器 ...
- linux 基本命令学习
原文: https://www.oschina.net/translate/useful-linux-commands-for-newbies?lang=chs&p=2