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 相关文章推荐
JavaScript OOP面向对象介绍
Dec 02 Javascript
extjs之去除s.gif的影响
Dec 25 Javascript
Javascript 实现复制(Copy)动作方法大全
Jun 20 Javascript
jQuery中index()方法用法实例
Dec 27 Javascript
JS遍历数组和对象的区别及递归遍历对象、数组、属性的方法详解
Jun 14 Javascript
基于js对象,操作属性、方法详解
Aug 11 Javascript
微信js-sdk分享功能接口常用逻辑封装示例
Oct 13 Javascript
微信小程序自定义底部导航带跳转功能
Nov 27 Javascript
JQuery表单元素取值赋值方法总结
May 12 jQuery
解决vue.js中settimeout遇到的问题(时间参数短效果不稳定)
Jul 21 Javascript
vue data对象重新赋值无效(未更改)的解决方式
Jul 24 Javascript
Vue按时间段查询数据组件使用详解
Aug 21 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数据类型之布尔型的介绍
2013/04/28 PHP
PHP的password_hash()使用实例
2014/03/17 PHP
在Thinkphp中使用ajax实现无刷新分页的方法
2016/10/25 PHP
PHP微信PC二维码登陆的实现思路
2017/07/13 PHP
Javascript 汉字字节判断
2009/08/01 Javascript
jquery组件使用中遇到的问题整理及解决
2014/02/21 Javascript
利用JavaScript的AngularJS库制作电子名片的方法
2015/06/18 Javascript
jQuery实现折叠、展开的菜单组效果代码
2015/09/16 Javascript
Vue.js每天必学之过滤器与自定义过滤器
2016/09/07 Javascript
js获取指定字符前/后的字符串简单实例
2016/10/27 Javascript
JS正则表达式完美实现身份证校验功能
2017/10/18 Javascript
150行Node.js实现的dns代理工具
2019/08/02 Javascript
vue实现输入框的模糊查询的示例代码(节流函数的应用场景)
2019/09/01 Javascript
深入学习Vue nextTick的用法及原理
2019/10/08 Javascript
js实现简单掷骰子小游戏
2019/10/24 Javascript
[02:12]2015国际邀请赛 SHOWOPEN
2015/08/05 DOTA
Python常用知识点汇总
2016/05/08 Python
Python pygorithm模块用法示例【常见算法测试】
2018/08/16 Python
解决python xlrd无法读取excel文件的问题
2018/12/25 Python
python爬虫selenium和phantomJs使用方法解析
2019/08/08 Python
PyCharm更改字体和界面样式的方法步骤
2019/09/27 Python
Anaconda之conda常用命令介绍(安装、更新、删除)
2019/10/06 Python
python wav模块获取采样率 采样点声道量化位数(实例代码)
2020/01/22 Python
Django实现内容缓存实例方法
2020/06/30 Python
Django如何继承AbstractUser扩展字段
2020/11/27 Python
用python计算文件的MD5值
2020/12/23 Python
美体小铺英国官网:The Body Shop英国
2017/01/24 全球购物
英国最大的女性服装零售商:Dorothy Perkins
2017/03/30 全球购物
XML文档面试题
2015/08/05 面试题
会计应聘求职信范文
2013/12/17 职场文书
项目管理计划书
2014/01/09 职场文书
班级德育工作实施方案
2014/02/21 职场文书
警示教育活动总结
2014/05/05 职场文书
工程部岗位职责范本
2015/04/11 职场文书
php引用传递
2021/04/01 PHP
Nginx解决403 forbidden的完整步骤
2021/04/01 Servers