【摘要】 Angularjs1.X中两种不同的双向数据绑定聊聊 Angularjs1.x中那些活见鬼的事情。一. html与Controller中的双向数据绑定html-Controller的双向数据绑定,在开发中非常常见,也是Angularjs1.x的宣传点之一,使用中并没有太多问题。

Angularjs1.X中两种不同的双向数据绑定

聊聊 Angularjs1.x中那些活见鬼的事情。

一. html与Controller中的双向数据绑定

html-Controller的双向数据绑定,在开发中非常常见,也是Angularjs1.x的宣传点之一,使用中并没有太多问题。

1.1数据从html流向controller

也就是从视图层流向模型层,原生html中需要使用表单元素(例如input标签)来收集用户输入信息,Angularjs中通过在表单元素上使用ng-model标签,当用户输入信息时,同步将用户输入的信息赋值给controller中的变量:

<body ng-app="myApp"><div id="main" ng-controller="myCtrl"><p>改变输出值:</p><input type="text" ng-model="testInfo.content" ng-change="showInput()"></div><script src="./angular.min.js"></script><script>angular.module('myApp',[]).controller('myCtrl',['$scope',function($scope){$scope.showInput = function() {          console.log($scope.testInfo.content);}}]); </script></body>

在页面上输入1234567即可看到,每次在页面输入数字后,控制台输出的$scope,testInfo.content的值都和页面保持一致:

1.2 数据从controller流向html

也就是从模型层流向数据层,当controller中的数据模型变量发生变化后,Angularjs又会根据数据模型的值去改变ng-model指令绑定的表单元素的值,使用ng-bind指令也可以被动获得来自controller的数据流。

我们编写如下demo进行测试:

<body ng-app="myApp"><div id="main" ng-controller="myCtrl"><button ng-click="add()">+1</button><p>改变输出值:</p><input type="text" ng-model="testInfo.content"><p>使用ng-bind绑定的标签:</p><p ng-bind="testInfo.content"></p></div><script src="./angular.min.js"></script><script>angular.module('myApp', []).controller('myCtrl', ['$scope', function($scope) {            //初始化$scope.testInfo = {                content: 0}$scope.add = function () {$scope.testInfo.content += 1;               console.log($scope.testInfo.content);}}]);    </script></body>

demo中,每次点击+1按钮,$scope.testInfo.content的值会增加1,我们可以看到页面上的结果:

1.3 刷视图

来看看第一个活见鬼的例子,demo跟上面很类似,只是将鼠标点击触发的方式改成了定时器自动触发

<body ng-app="myApp"><div id="main" ng-controller="myCtrl"><button ng-click="add()">+1</button><p>改变输出值:</p><input type="text" ng-model="testInfo.content"><p>使用ng-bind绑定的标签:</p><p ng-bind="testInfo.content"></p></div><script src="./angular.min.js"></script><script>angular.module('myApp', []).controller('myCtrl', ['$scope', function($scope) {            //初始化$scope.testInfo = {                content: 0}            //定时自增setInterval(function () {$scope.testInfo.content += 1;              console.log('$scope.testInfo.content的值现在是:',$scope.testInfo.content);},1000)}]);    </script></body>

你会发现,数据模型一直在变,但是页面却没有刷新:

这里就是 Angularjs1.X双向数据绑定中的第一个坑 ,你会发现$scope上绑定的数据模型html中显示的内容有时候并不是实时关联的。这其实和Angularjs1.X的执行机制有关系。

如果我们自己来考虑,javascript中有一个变量的值发生了变化,现在要将这个值同步到html页面上,需要怎么做呢?我们需要获取到这个DOM元素,然后改变它的innerHTML属性,如果是表单元素就修改value。其实Angularjs也是这样做的,只不过使用了自己的封装的方法——$apply()。那么此处的问题其实就在于,在setInterval的回调函数中去修改数据模型的值时,没有触发$apply()方法来更新视图,而通过调用Angularjs封装的ng-*方法(例如ng-click点击方法)来修改视图模型时,会自动触发$apply()方法,视图也就同步刷新了。

  • 解决方案1

    使用Angularjs封装过的$interval服务来实现定时任务,感兴趣的读者可以自己看一下Angularjs源码中$intervalProvider的部分,就会发现在方法最后的地方调用了$rootScope.$apply()

  • 解决方案2

    如果依然使用javascript原生的定时方法,那么则需要在修改完视图的数据模型后,手动调用$scope.$apply()方法来将数据模型的变动同步到html页面中。

二. Controller与Directive中的双向数据绑定

除了controller与html中的双向绑定,Angularjs中还有另一个双向数据绑定,那就是controller与directive之间的绑定。绑定的形式有很多种,我们先来看一下最常见的双向绑定

2.1 directive中的双向数据绑定

在设定自定义指令的scope参数时,将属性的值设置为=就可以实现双向数据绑定,这里API的解释是:

父级controller中的指定变量会与自定义指令link函数中的变量相互影响

下面的实例中,我们将看看controller中的数据模型$scope.testInfo.content的值与自定义指令中scope.pagination如何相互影响,是否如定义所说这里的绑定真的是双向的。示例界面如下(demo源码请见附件demo.html文件):

  • 每次点击+1按钮,Scope.testInfo.content的值都会增加1

  • 每次点击show $scope.testInfo,控制台都会打印出$scope.testInfo的值

  • 每次点击标签上的数字,则会打印出自定义指令中scope.pagination的值,并将该值进行自增

接下来的测试操作,我们将按照如下的流程进行:

  1. 点击5次+1按钮,再点击5次数字标签

  2. 点击show $scope.testInfo按钮

2.2 怎么又不刷新了

随着上一节的操作步骤,我们一起来见证双向数据绑定中又一次闹鬼事件:

  • 点击5次+1按钮,再点击5次数字标签

    结果为:

  • 我们看到,第一次点击数字标签时,控制台打出了link函数中scope.pagination的值为5,这说明$scope.testInfo.content的值被传递给了自定义指令中的scope.pagination,也就是说数据从controller流向了directive。而当我们再点击4次数字标签(一共点了5次)后,从控制台可以看出,scope.pagination的值已经成为10,而页面上使用ng-bind指令获取到的结果却依旧是5。也就是说,数据从没有从directive流向controller。是不是有一种被骗的感觉?别着急,接着看。

  • 点击show $scope.testInfo按钮

    结果为:

当我们点击show $scope.testInfo时,控制台打印出了$scope.testInfo.content的值为5,这下证据坐实了,明明说好的双向数据绑定,然而当自定义指令中的scope.pagination改变时,$scope.testInfo.content并没有跟着一起改变。But!!!!我们会发现,这个show $scope.testInfo点下去以后,页面上通过ng-bind绑定的值却变成了10。也就是说,数据又从directive流回了controller

官方建议使用$watch方法来追踪scope中的变量,而当我们这样做时,会发现$watch函数仅能追踪到那些通过修改controller中的数据模型而影响link函数中变量的行为并更新视图

这里就是 Angularjs1.X双向数据绑定中的第二个坑,controller和directive中所谓的双向数据绑定,并不能追踪指定变量的所有变化,而且不是同步完成的。

其实这里的问题仍然和Angularjs的运行机制有关,解决方案如下:

  • 解决方案1

    使用自定义指令的templateUrl属性替换当前指令的模板,使用ng-click指令来绑定一个点击响应函数,在响应函数中改变scope.piganation的值。

  • 解决方案2

    在手动绑定的监听回调中,修改自定义指令作用域内的变量后,使用scope.$emit( )方法通知其父级controller,并在controller中使用$scope.$on( )方法监听同名事件,并修改对应的数据模型的值。

  • 解决方案3

    每当改变自定义指令中的变量值后,调用scope.$apply()方法,将directive中的变量值同步至controller的数据模型以及页面。

三.原理和实战总结

3.1 Angularjs中双向数据绑定的基本原理

Angularjs中的双向数据绑定,是通过一种叫做*"脏循环检查(dirty-checking)"的机制实现的。

其基本过程是这样的,每当我们使用ng-modelng-bind指令将数据模型中的某个变量值和html页面上某个标签的内容联系起来时,Angular就会把这些变量放进一个WatchCollection的集合中,并自动帮我们来监控这些变量。每当WatchCollection中有变量出现变动时,Angular就会遍历WatchCollection来查看是否有其他监控中的变量也被影响,每当有一个变量被影响,Angular都会在遍历后再进行一次遍历,直到某一次遍历后WatchCollection中的变量都没有变化,则Angular会认为当前的改动已经稳定了,然后才会将数据模型的变化同步到DOM元素上去,也就实现了数据绑定。

我们可以把WatchCollection理解为当前页面的一种抽象,其中包含着页面上所有有可能发生变化的部分。

3.2 双向数据绑定的实践经验

想要在Angularjs项目中更加稳定地使用双向数据绑定,笔者的建议是:

Angularjs项目中,尽可能地使用Angular告诉你的方式去编写所希望实现的功能。

我们可以回顾一下上面在使用双向数据绑定发生异常时的场景:

  • 使用了原生的定时器(Angular中你应该使用$interval,$timeout服务)

  • 用类原生方法(bind)为元素添加事件监听器,并在回调函数中修改了变量的值(Angular中,你应该使用ng-click来实现点击事件的监听)

  • ...

你会发现,每当自己没有按照Angular的方式去编写代码,或者没有按照一个模块设计的初衷去使用它时,就无法确切地得到期望的结果。这是很容易理解的,如果你没有按照Angular要求的方式书写代码,凭什么期望它对你的代码做出100%正确的回应呢?至于上述两种数据绑定中出现问题的解决方案,上文已经有所提及,此处不再赘述。

许多人都听说过"尽量不要在controller中操作DOM"这句话,实际上它并不意味着你在controller中操作DOM会导致程序报错,而是在说如果你同时使用jQueryAngular两套系统来管理自己的代码,但又没有按照官方指定的方式来规避它们之间的冲突,那代码很可能会变得不稳定。想想当年腾讯电脑管家360安全卫士将你的电脑卡死的场景,你就明白这样做的结果了。

四. 小结——所谓高手

笔者曾经看过这样一段话,觉得深有感触:

所谓高手,是指那些熟知套路创意无穷的人。而高手之间的较量,归根结底都是基本功的比拼。

愿有朝一日,你也能成为高手。

来源:华为云社区原创  作者:大史不说话

Angularjs进阶笔记(1)—不同类型的双向数据绑定相关推荐

  1. Angularjs进阶笔记(2)—自定义指令中的数据绑定

    [摘要]有关自定义指令的scope参数,网上很多文章都在讲这3种绑定方式实现的效果是什么,但几乎没有人讲到底怎么使用,本篇希望聊聊到底怎么用这个话题. 一. 自定义指令 自定义指令,是Angularj ...

  2. Build Your Own Angularjs 读书笔记(AngularJS牛逼的地方在于它内嵌了一个表达式到Function对象的编译器。。。当然还有DI框架)

    Build Your Own Angularjs 读书笔记 目录 [隐藏] 1 项目配置 2 作用域 3 表达式与过滤器 4 模块与依赖注入 5 辅助函数 6 指令 项目配置[编辑] npm pack ...

  3. Android进阶笔记:Messenger源码详解

    Messenger可以理解为一个是用于发送消息的一个类用法也很多,这里主要分析一下再跨进程的情况下Messenger的实现流程与源码分析.相信结合前面两篇关于aidl解析文章能够更好的对aidl有一个 ...

  4. AngularJs学习笔记--Forms

    控件(input.select.textarea)是用户输入数据的一种方式.Form(表单)是这些控件的集合,目的是将相关的控件进行分组. 表单和控件提供了验证服务,所以用户可以收到无效输入的提示.这 ...

  5. AngularJS学习笔记(一)

    前言 几个月之前了解过一点Angular,主要是通过phonecat应用程序了解一些入门东西,但是当被问及什么是Angular或者你对Angular的理解时,只记得一个MVVM双向数据绑定,显然这是不 ...

  6. Shell 编程进阶笔记

    这几篇博文主要记录博主的Linux 学习之路,用作以后回顾和参考.大家可以选择略过也可以作参考. (一)Linux 初步笔记 (二)Linux 进阶笔记(一) (三)Linux 进阶笔记(二) (四) ...

  7. ggplot2箱式图两两比较_R语言进阶笔记2 | 长数据与ggplot2

    1. 长数据是什么鬼? 之前介绍了如何将多个性状的箱线图放在一个图上,比如learnasreml包中的fm数据,它有h1~h5五年的株高数据,想对它进行作图. 「数据预览:」 > library ...

  8. AngularJs 在ng-repeat中动态使用ng-model进行双向数据绑定(二)

    这次来些稍微复杂点的功能需求: 1.在info旁边的输入框中输入数字,根据数字的多少来动态显示这部分输入框: 其中,check部分的输入框初始是没有的(当然,也可以有!): 2.每点击一次增加按钮,下 ...

  9. Android 进阶笔记,包含常用的技术框架、博客社区、书籍等。

    AndroidNote 项目地址:venshine/AndroidNote 简介:Android 进阶笔记,包含常用的技术框架.博客社区.书籍等. 大纲 技术框架 图片加载 网络请求 数据库 ORM ...

最新文章

  1. SourceChangeWarning:验证集上准确率很高,但是测试集上很低
  2. es 插入数据_记录一次Java导入百万级数据到Elasticsearch经历
  3. 单片机光敏电阻控制蜂鸣器_走进单片机|第七期:湿度检测
  4. 程序员编程艺术:第二章、字符串是否包含问题
  5. 黑科技抢先尝(续2) - Windows terminal中Powershell Tab的美化全攻略
  6. 倒计时按钮_办公小技巧:轻松玩转PPT秒针倒计时
  7. 同步方法中的锁对象_互斥锁与读写锁:如何使用锁完成Go程同步?
  8. 手机MODEM 开发(28)--- VoLTE介绍
  9. 大数据在各行业中的应用表现
  10. Atitit 单项功能开发 最佳实践规范 标准化流程attilax总结.docx
  11. 大学生IT网络创业计划书
  12. python 打包exe_python打包exe能运行但是没有结果解决方案
  13. 写一本技术书能赚多少钱
  14. 模糊综合评价在实际问题中的应用(案例)
  15. rtthread spiflash (w25q64)
  16. 【C语言】C语言之数字金字塔全家桶
  17. robotframework之解决导入httplibrary库一直标红的问题
  18. YOLOV5使用过程中可能出现的问题及解决方法
  19. matlab 不规则 griddata,MATLAB 不规则随机数据点,画三维曲面图形,griddata, meshgrid...
  20. 牛客网 小睿睿的方案 解题报告

热门文章

  1. python apscheduler 动态_基于Flask-APScheduler实现添加动态定时任务
  2. php 5.4 安装xcache,安装xcache为你的PHP加速
  3. 构造函数能默认初始化其静态成员么?
  4. dirname,basename的用法与用途
  5. 目标检测之Faster-RCNN的pytorch代码详解(模型准备篇)
  6. 51nod 1686 第K大区间 二分瞎搞
  7. 第三章 springboot + jedisCluster(转载)
  8. 面向对象与原型(二)
  9. leetcode 746 python 动态规划
  10. 計算機二級-java06