AngularJS控制器controller正确的通信的方法


Posted in Javascript onJanuary 25, 2016

AngularJS是一款非常强大的前端MVC框架,AngularJS中的controller是个函数,用来向视图的作用域($scope)添加额外的功能,我们用它来给作用域对象设置初始状态,并添加自定义行为。

当我们在创建新的控制器时,angularJS会帮我们生成并传递一个新的$scope对象给这个controller,在angularJS应用的中的任何一个部分,都有父级作用域的存在,顶级就是ng-app所在的层级,它的父级作用域就是$rootScope。

每个$scope的$root指向$rootScope, $cope.$parent指向父级作用域。

cotroller之间的通信本质上是当前的controller所在的$scope如何跟其他controller上的$scope进行通信。

通常有3中解决方式:

利用作用域继承的原理,子控制器访问父级控制器中的内容。 使用angularJS中的事件,也就是使用$on,$emit,$broadcast进行消息传递 使用angularJS中的服务
第一种方式

即作用域嵌套作用域,有一定的使用限制,需要作用域嵌套起来,在实际开发中这种场景相对比较少,但也不是没有,这种方式更简单直接。

angularJS中默认情况下,当前作用域中无法找到某个属性时,就会在父级作用域中进行查找,若找不到直至查找到$rootScope。 如果在$rootScope中也无法找到程序依旧运行,但视图不会更新。

示例

javascript

//Javascript
app.controller('ParentController', function($scope) { 
$scope.person = {greeted: false};
});
app.controller('ChildController', function($scope) {
$scope.sayHello = function() {
$scope.person.name = 'Ari Lerner';
};
});
//HTML
<div ng-controller="ParentController">
<div ng-controller="ChildController">
<a ng-click="sayHello()">Say hello</a>
</div>
{{ person }}
</div>
//result
{"greeted":true, "name": "Ari Lerner"}

第二种方式

因为作用域是有层次的,所以可以利用作用域链传递事件。

传递事件有2种方式: * $broadcast: 触发的事件要通知整个事件系统(允许任意作用域处理这个事件)就要向下传播。 * $emit: 如果要提醒一个全局模块,需要通知更高层次的作用域时(例如$rootscope)需要把事件向上传递。

作用域上使用$on进行事件监听。

示例

JavaScript

app.controller('ParentController', function($scope) { 
$scope.$on('$fromSubControllerClick', function(e,data){
console.log(data); // hello
});
});
app.controller('ChildController', function($scope) {
$scope.sayHello = function() {
$scope.$emit('$fromSubControllerClick','hello');
};
});
//HTML
<div ng-controller="ParentController">
<div ng-controller="ChildController">
<a ng-click="sayHello()">Say hello</a>
</div>
</div>

在这里想要说的另外一个问题就是事件传播的性能问题,$broadcast+$on的方式回通知所有的子作用域,这里就会有性能问题,所以推荐使用$emit+$on的方式,为了进一步提升性能,定义的事件处理函数要在作用域销毁时一起释放掉。

使用$emit+$on的方式需要我们将事件监听绑定在$rootScope上,例如:

JavaScript

angular
.module('MyApp')
.controller('MyController', ['$scope', '$rootScope', function MyController($scope, $rootScope) {
var unbind = $rootScope.$on('someComponent.someCrazyEvent', function(){
console.log('foo');
});
$scope.$on('$destroy', unbind);
}
]);

但是这种方式有点繁琐,定义多个事件处理函数时整个人都不好了,所以我们来改进一下

利用装饰器来定义一个新的事件绑定函数:

JavaScript

angular
.module('MyApp')
.config(['$provide', function($provide){
$provide.decorator('$rootScope', ['$delegate', function($delegate){
Object.defineProperty($delegate.constructor.prototype, '$onRootScope', {
value: function(name, listener){
var unsubscribe = $delegate.$on(name, listener);
this.$on('$destroy', unsubscribe);
return unsubscribe;
},
enumerable: false
});
return $delegate;
}]);
}]);

那么我们在控制器中定义事件处理函数时:

JavaScript

angular
.module('MyApp')
.controller('MyController', ['$scope', function MyController($scope) {
$scope.$onRootScope('someComponent.someCrazyEvent', function(){
console.log('foo');
});
}
]);

个人强烈推荐此种做法

第三种方式

利用angularJS中service单例模式的特性,服务(service)提供了一种能在应用的整个生命周期内保持数据的方式,能够在控制器之间进行通信,且能保证数据的一致性。

一般我们都会封装server来为应用提供访问数据的接口,或者跟远程进行数据交互。

示例

JavaScript

var myApp = angular.module("myApp", []);
myApp.factory('Data', function() {
return {
name: "Ting"
}
});
myApp.controller('FirstCtrl', function($scope, Data) {
$scope.data = Data;
$scope.setName = function() {
Data.name = "Jack";
}
});
myApp.controller('SecondCtrl', function($scope, Data) {
$scope.data = Data;
$scope.setName = function() {
Data.name = "Moby";
}
});

以上所述是针对AngularJS控制器controller正确的通信的方法,希望能够帮助到大家。

Javascript 相关文章推荐
更快的异步执行(setTimeout多浏览器)
Aug 12 Javascript
jQuery插件Slider Revolution实现响应动画滑动图片切换效果
Jun 05 Javascript
javascript单例模式的简单实现方法
Jul 25 Javascript
在Mac OS上安装使用Node.js的项目自动化构建工具Gulp
Jun 18 Javascript
CSS+jQuery实现简单的折叠菜单
Dec 20 Javascript
在一个页面重复使用一个js函数的方法详解
Dec 26 Javascript
vue中SPA单页面应用程序详解
Nov 07 Javascript
JS基于开关思想实现的数组去重功能【案例】
Feb 18 Javascript
Vuex持久化插件(vuex-persistedstate)解决刷新数据消失的问题
Apr 16 Javascript
vue+AI智能机器人回复功能实现
Jul 16 Javascript
如何手动实现一个 JavaScript 模块执行器
Oct 16 Javascript
Nuxt的路由动画效果案例
Nov 06 Javascript
AngularJS入门心得之directive和controller通信过程
Jan 25 #Javascript
详解jquery事件delegate()的使用方法
Jan 25 #Javascript
AngularJS中的Directive实现延迟加载
Jan 25 #Javascript
AngularJS中的Directive自定义一个表格
Jan 25 #Javascript
理解JavaScript事件对象
Jan 25 #Javascript
AngularJS中如何使用$parse或$eval在运行时对Scope变量赋值
Jan 25 #Javascript
学习JavaScript事件流和事件处理程序
Jan 25 #Javascript
You might like
php连接odbc数据源并保存与查询数据的方法
2014/12/24 PHP
YII CLinkPager分页类扩展增加显示共多少页
2016/01/29 PHP
thinkphp3.x中cookie方法的用法分析
2016/05/19 PHP
php根据数据id自动生成编号的实现方法
2016/10/16 PHP
PHP获取当前日期及本周一是几月几号的方法
2017/03/28 PHP
使用PHP json_decode可能遇到的坑与解决方法
2017/08/03 PHP
PHP常用函数之base64图片上传功能详解
2019/10/21 PHP
TP5框架页面跳转样式操作示例
2020/04/05 PHP
javascript之通用简单的table选项卡实现(二)
2010/05/09 Javascript
jQuery 源码分析笔记(5) jQuery.support
2011/06/19 Javascript
JS记录用户登录次数实现代码
2014/01/15 Javascript
js简单的弹出框有关闭按钮
2014/05/05 Javascript
JQuery中基础过滤选择器用法实例分析
2015/05/18 Javascript
JavaScript数组方法总结分析
2016/05/06 Javascript
BootStrap 智能表单实战系列(五) 表单依赖插件处理
2016/06/13 Javascript
jQuery实现鼠标滑过图片移动特效
2016/12/08 Javascript
微信小程序登录数据解密及状态维持实例详解
2019/05/06 Javascript
node.js Promise对象的使用方法实例分析
2019/12/26 Javascript
tracking.js实现前端人脸识别功能
2020/04/16 Javascript
24个ES6方法解决JS实际开发问题(小结)
2020/05/31 Javascript
通过实例了解JS执行上下文运行原理
2020/06/17 Javascript
python二叉树遍历的实现方法
2013/11/21 Python
Python中函数的用法实例教程
2014/09/08 Python
Python中的连接符(+、+=)示例详解
2017/01/13 Python
Python生成随机数组的方法小结
2017/04/15 Python
深入浅析Python传值与传址
2018/07/10 Python
Python学习笔记之Zip和Enumerate用法实例分析
2019/08/14 Python
python 有效的括号的实现代码示例
2019/11/11 Python
Python Pandas 转换unix时间戳方式
2019/12/07 Python
CSS中越界问题的经典解决方案【推荐】
2016/04/19 HTML / CSS
雷曼兄弟的五金店:Lehman’s Hardware Store
2019/04/10 全球购物
AVI-8手表美国官方商店:AVI-8 USA
2019/04/10 全球购物
2014年国庆节演讲稿
2014/09/02 职场文书
观看建国大业观后感
2015/06/01 职场文书
2016新教师培训心得体会范文
2016/01/08 职场文书
《作风建设永远在路上》心得体会
2016/01/21 职场文书