详解JavaScript的AngularJS框架中的表达式与指令
“指令属性”就是绑定在dom元素上的函数,它可以调用方法、定义行为、绑定controller及$scope对象、操作dom,等等等等。
当浏览器启动、开始解析html(像平时一样)时,dom元素上的指令属性就会跟其他属性一样被解析。
当一个angular.js应用启动,angular编译器就会遍历dom树(从有ng-app指令属性的那个dom元素开始,如我们在本系列第一篇里所提过的),解析html,寻找这些指令属性函数。
当在一个dom元素上找到一个或多个这样的指令属性函数,它们就会被收集起来、排序,然后按照优先级顺序被执行。
每个指令属性都有自己的优先级,在我们关于指令属性的专题文章里(http://www.ng-newsletter.com/posts/directives.html),你可以找到更深入的信息。
angular.js应用的动态性和响应能力,都要归功于指令属性。之前我们已经看过一些指令属性的用例:
ng-model
<input ng-model="name" name="name" placeholder="enter your name"/> <h4>your name: {{ name }}</h4>
试试看
ng-model指令属性(我们在之前的章节使用过它),被用来将dom文本输入框的值,跟controller里的$scope model绑定起来。具体的实现过程,是在这个值上绑定了一个$watch函数(类似javascript里的事件监听函数)。
$watch函数(在使用时)运行在angular.js的事件循环(即$digest循环)里,让angular.js能够对dom进行相应的更新。请关注我们关于$digest循环的高级文章!
在angular.js应用的开发中,我们用指令属性来将行为绑定到dom上。指令属性的使用,是一个 应用能否拥有动态性、响应能力的关键。angular.js提供了很多缺省的指令属性,现在让我们来看看其中几个,以及如何使用它们。
几个常见的指令属性:
{{ 表达式 }}
这个双大括号指令属性,使用$watch()函数,给括号内的表达式注册了一个监听器。正是这个$watch函数,让angular.js能够实时自动更新view。
那么,到底什么算是个表达式?
要想理解指令属性的运作,我们必须先理解表达式,所以这里我们就绕个路,看看本系列的第五部分——表达式。在之前的例子里我们已经见过表达式,例如 {{ person.name }} 和 {{ clock }}。
{{ 8 + 1 }}9 {{ person }}{"name":"ari lerner"} {{ 10 * 3.3 | currency }}$33.00
最后的例子里 (10 * 3.3 | currency) 用了一个过滤器。本系列之后的部分,会深入介绍过滤器。
表达式粗略来看有点像 eval(javascript) 的结果。它们会经过angular.js的处理,从而拥有以下重要而独特的性质:
所有表达式都在scope这个context里被执行,因此可以使用所有本地 $scope 中的变量。
如果一个表达式的执行导致类型错误或引用错误,这些错误将不会被抛出。
表达式里不允许任何控制函数流程的功能(如if/else等条件语句)
表达式可接受一个或多个串联起来的过滤器。
试试看
试试输入“person“,“clock“或其他数学算式如2+4。你甚至可以操作scope,例如,试试输入person.name = ”ari”; person.age = 28; person 或 clock
表达式都运行在调用它们的scope里,所以一个表达式可访问并操作其scope上的一切。由此,你可以使用表达式遍历其scope的属性(我们在ng-repeat中会看到这一应用)、调用scope里的函数,或者对scope中的变量进行数学运算。
现在,让我们回到指令属性…
ng-init
ng-init指令属性是一个在启动时运行的函数(在程序进入运行阶段之前)。它让我们能够在程序运行前设定初始变量的值:
<b ng-init='name = "ari lerner"'>hello, {{ name }}</b>
试试看
hello, ari lerner
ng-click
ng-click指令属性给dom元素注册了一个点击事件的监听器。当此dom元素上有点击事件发生(即当此button或link被点击时),angular.js就会执行表达式的内容,并相应地更新view。
<button ng-click="counter = counter + 1">add one</button> current counter: {{ counter }}
试试看
我们也可以用ng-click 来调用在controller里写好并绑定在$scope上的函数,例如:
<button ng-click="sayhello()">say hello</button>
controller 里的函数:
app.controller('mycontroller', function($scope) { $scope.sayhello = function() { alert("hello!"); } });
试试看
ng-show / ng-hide
ng-show和ng-hide指令,根据赋予它们的表达式的值的真假性(truthy),来显示和隐藏它们所属的那一部分dom。
我们在这里不会深入,但你应该熟悉javascript中变量值的“truthy”和“falsy”概念。
<button ng-init="shouldshow = true" ng-click="shouldshow = !shouldshow">flip the shouldshow variable</button> <div ng-show="shouldshow"> <h3>showing {{ shouldshow }}</h3> </div> <div ng-hide="shouldshow"> <h3>hiding {{ shouldshow }}</h3> </div>
ng-repeat
ng-repeat指令遍历一个数据集合中的每个数据元素,加载html模版把数据渲染出来。被重复使用的模版元素,就是我们绑定了这个指令属性的dom元素。每一个使用模版渲染的dom元素都有自己的scope。
在更多的解释之前,我们先看一个例子。假设我们的controller里有这样一个数据元素的数组:
$scope.roommates = [ { name: 'ari'}, { name: 'q'}, { name: 'sean'}, { name: 'anand'} ];
在view里我们可以用ng-repeat来遍历这个集合里的数据:
<ul> <li ng-repeat="person in roommates">{{ person.name }}</li> </ul>
对赋予ng-repeat的表达式稍作改动,我们还可以用它遍历一个由成对的key-value数据组成的集合。例如,假设我们有一个人名和他们最喜欢的颜色的数据集合:
$scope.people = { 'ari': 'orange', 'q': 'purple', 'sean': 'green' }
要遍历它,我们可以给ng-repeat指令属性赋予这个表达式: (key, value) in object:
<ul> <li ng-repeat="(name, color) in people">{{ name }}'s favorite color is {{ color }} </li> </ul>
推荐阅读
-
AngularJS的ng-repeat指令与scope继承关系实例详解
-
AngularJS框架中的双向数据绑定机制详解【减少需要重复的开发代码量】
-
JavaScript中的垃圾回收与内存泄漏示例详解
-
AngularJS框架的ng-app指令与自动加载实现方法分析
-
理解 javascript 中的函数表达式与函数声明
-
JavaScript中构造函数与原型链之间的关系详解
-
JavaScript编程中window的location与history对象详解
-
JavaScript中class类的静态方法、普通方法与构造方法详解
-
Javascript中this关键字指向问题的测试与详解
-
详解JavaScript的AngularJS框架中的表达式与指令