AngularJS框架中的数据绑定是如何工作的?
我还没有在他们的网站上找到技术细节。当数据从视图传播到模型时,它是如何工作的,这或多或少是清楚的。但是AngularJS如何在没有setter和getter的情况下跟踪模型财产的更改?
我发现有JavaScript观察者可能会做这项工作。但Internet Explorer 6和Internet Explorer 7不支持它们。那么AngularJS如何知道我更改了例如以下内容,并将此更改反映在视图上?
myobject.myproperty="new value";
用图片解释:
数据绑定需要映射
作用域中的引用与模板中的引用不完全相同。当您数据绑定两个对象时,需要第三个对象侦听第一个对象并修改另一个对象。
在这里,当您修改<input>时,您会触摸data-ref3。经典的数据绑定机制将改变data-ref4。那么其他{{数据}}表达式将如何移动?
事件导致$digest()
Angular维护每个绑定的oldValue和newValue。在每一个Angular事件之后,著名的$digest()循环都会检查WatchList以查看是否发生了变化。这些Angular事件是ng单击、ng更改、$http完成。。。只要任何oldValue与newValue不同,$digest()就会循环。
在上一张图片中,它将注意到data-ref1和data-ref2已更改。
结论
这有点像鸡蛋和鸡肉。你永远不知道是谁开始,但希望大多数时候都能像预期的那样。
另一点是,您可以很容易地理解简单绑定对内存和CPU的影响。希望台式机足够胖,可以处理这个问题。手机没有那么强。
数据绑定:
什么是数据绑定?
每当用户更改视图中的数据时,范围模型中就会出现该更改的更新,反之亦然。
怎么可能?
简短回答:借助消化循环。
说明:Angular js在作用域模型上设置观察者,如果模型发生变化,它会触发监听器函数。
$scope.$watch('modelVar' , function(newValue,oldValue){
//使用新值更新Dom代码
});
那么何时以及如何调用观察程序函数?
作为摘要循环的一部分调用观察程序函数。
摘要循环是作为angular js内置指令/服务的一部分自动触发的,如ng model、ng bind、$timeout、ng click等。。让你触发消化周期。
摘要循环功能:
$scope.$digest() -> digest cycle against the current scope.
$scope.$apply() -> digest cycle against the parent scope
即$rootScope.$apply()
注:$apply()等于$rootScope$digest()这意味着脏检查从根或顶部或父作用域开始,一直到angular js应用程序中的所有子$scopes。
上述功能适用于上述版本的浏览器IE,只需确保您的应用程序是angular js应用程序,这意味着您使用的是脚本标记中引用的angularjs框架脚本文件。
非常感谢。
Misko已经很好地描述了数据绑定的工作方式,但我想补充一下我对数据绑定性能问题的看法。
正如Misko所说,大约2000个绑定是您开始发现问题的地方,但无论如何,页面上的信息不应该超过2000条。这可能是正确的,但并不是每个数据绑定都对用户可见。一旦开始使用双向绑定构建任何类型的小部件或数据网格,就可以很容易地达到2000个绑定,而不会出现糟糕的UX。
例如,考虑一个组合框,您可以在其中键入文本以筛选可用选项。这种控件可能有150个项目,并且仍然非常有用。如果它有一些额外的特性(例如,当前所选选项上的特定类),则开始为每个选项提供3-5个绑定。在一个页面上放置三个小部件(例如,一个用于选择国家,另一个用于在所述国家中选择城市,第三个用于选择酒店),您已经绑定了1000到2000个绑定。
或者考虑企业web应用程序中的数据网格。每页50行不是不合理的,每行可以有10-20列。如果您使用ng重复进行构建,和/或在某些使用某些绑定的单元格中有信息,那么仅使用此网格就可能接近2000个绑定。
在使用AngularJS时,我发现这是一个巨大的问题,到目前为止,我能找到的唯一解决方案是在不使用双向绑定的情况下构造小部件,而不是使用ngOnce、注销观察者和类似的技巧,或者构造用jQuery和DOM操纵构建DOM的指令。我觉得这一点一开始就违背了使用Angular的目的。
我很想听听关于处理这个问题的其他方法的建议,但也许我应该自己写一个问题。我想在评论中发表这一点,但事实证明这太长了。。。
TL;博士数据绑定可能会导致复杂页面的性能问题。
AngularJs支持双向数据绑定。意味着您可以访问数据视图->控制器和控制器->视图
例如。
1)
// If $scope have some value in Controller.
$scope.name = "Peter";
// HTML
<div> {{ name }} </div>
O/P
Peter
您可以在ng模型中绑定数据,如:-2)
<input ng-model="name" />
<div> {{ name }} </div>
在上面的示例中,无论用户将给出什么输入,它都将在<div>标记中可见。
如果要将输入从html绑定到控制器:-3)
<form name="myForm" ng-submit="registration()">
<label> Name </lbel>
<input ng-model="name" />
</form>
如果您想在控制器中使用输入名称,
$scope.name = {};
$scope.registration = function() {
console.log("You will get the name here ", $scope.name);
};
ng模型绑定我们的视图并在表达式{{}}中呈现它。ng模型是在视图中显示给用户并与用户交互的数据。因此,很容易在AngularJs中绑定数据。