Angular按照发布/订阅模式设计了其事件系统,使用时需要“发布”事件,并在适当的位置“订阅”或“退订”事件,就像邮箱里面大量的订阅邮件一样,当我们不需要时就可以将其退订了。具体到开发中,对应着$scope和$rootScope的$emit$broadcast$on方法。本文介绍Angular的事件机制,包括$scope和$rootScope处理事件上的异同,$broadcast、$emit和$on的使用方式及他们区别等内容。

$scope与$scope之间的关系,$scope与$rootScope之间的关系

要理解Angular的事件机制,首先需要了解$scope$scope之间的关系以及$scope$rootScope之间的关系。$rootScope是唯一真神,是万域起源,是所有$scope的最终祖先。而$scope$scope之间可能的关系包括父子关系和兄弟关系。还记得controller之间的关系吗,Angular为每个controller分配一个独立的$scope,controller之间的关系也对应着$scope之间的关系:

<div ng-controller="ParentCtrl as parent">
{{ parent.data }}
<div ng-controller="SiblingOneCtrl as sib1">
{{ sib1.data }}
</div>
<div ng-controller="SiblingTwoCtrl as sib2">
{{ sib2.data }}
</div>
</div>

发布、订阅、退订

$broadcast$emit用于发布事件,他们将事件名称和事件内容发布出去,就像是高考榜单一样,事件名称相当于考生的名字,而事件内容相当于考生的成绩等信息:

$scope.$broadcast('EVENT_NAME', 'Data to send');
$scope.$emit('EVENT_NAME', 'Data to send');

$on用于订阅事件,事件名称是订阅的唯一标识,每个考生看榜单时都要寻找自己的名字,然后根据自己的成绩等信息决定下一步应该报考什么学校:

$scope.$on('EVENT_NAME', function(event, args) {
// balabala
});

Angular的退订事件有些奇怪,并没有类似于其他语言的$off方法,所以不要想当然的按照如下方式进行事件的退订操作:

// 不要这样做
$scope.$off('EVENT_NAME');

事实上,Angular的事件退订方法隐藏在事件订阅里面:使用$on订阅事件时会返回一个函数,而此函数就是用来退订事件的方法,就像是考生看到了自己的成绩后禀告父母大人,“商量着”选取学校填报志愿,而此志愿单就是结束整个高考榜单的结束:

// 订阅事件返回用于退订事件的函数
var deregister = $scope.$on('EVENT_NAME', function(event, args) {
// balabala
}); // 退订事件
deregister();

$broadcast相当于战斗机轰炸,$emit相当于射箭

$broadcast$emit都用于发布事件,但从名字就可以看出他们的不同点:$broadcast是自上而下的广播,所有能听到的都可以对其进行反应。而$emit是自下而上的射箭,只有在箭矢的轨迹上才能对其做出反应。

具体到Angular上,即从一个$scope上通过$broadcast发布的事件,他的所有后代$scope都可以对此事件做出响应:

// 父$scope通过$broadcast发布事件
app.controller('ParentCtrl', ['$scope', function($scope) {
$scope.$broadcast("parent", 'Data to Send');
}])
//所有子$scope都可以通过$on订阅事件
.controller('SiblingOneCtrl', ['$scope', function($scope) {
$scope.$on("parent", function(event, 'Data to Send') {
// balabala
});
}])
.controller('SiblingTwoCtrl', ['$scope', function($scope) {
$scope.$on("parent", function(event, 'Data to Send') {
// balabala
});
}]);

而通过$emit发布的事件,只有他的祖先$scope可以做出响应,并且其中任一祖先都可以将此事件终结掉,不让其继续传播:

// 子$scope通过$emit发布事件
app.controller('SiblingOneCtrl', ['$scope', function($scope) {
$scope.$emit("sib1", 'Data to Send');
}])
// 父$scope通过$on订阅事件
.controller('ParentCtrl', ['$scope', function($scope) {
$scope.$on("sib1", function(event, 'Data to Send') {
// balabala
});
}])
// 其兄弟$scope对其$emit的事件一无所知,所以不能订阅其事件
.controller('SiblingTwoCtrl', ['$scope', function($scope) {
// 不要这样做
$scope.$on("sib1", function(event, 'Data to Send') {
// balabala
});
}]);

$emit发布事件的响应道路上,其任一祖先如果感觉不再需要此事件了,就可以通过如下方式终结此事件:

app.controller('ParentCtrl', ['$scope', function($scope) {
$scope.$on("sib1", function(event, 'Data to Send') {
// balabala
event.stopPropagation(); // 终止事件继续“冒泡”
});
}])

$rootScope的$broadcast和$emit

上面说过$rootScope是所有$scope的最终祖先,所以通过$rootScope$broadcast发布的事件可以被所有$scope接收到,包括$rootScope

app.controller('SomeCtrl', ['$rootScope', function($rootScope) {
$rootScope.$broadcast("rootEvent", 'Data to Send'); // $rootScope也可以通过$on订阅从$rootScope.$broadcast发布的事件
$rootScope.$on("rootEvent", function(event, 'Data to Send') {
// balabala
});
}])
// 所有$scope都能够通过$on订阅从$rootScope.$broadcast发布的事件
.controller('ParentCtrl', ['$scope', function($scope) {
$scope.$on("rootEvent", function(event, 'Data to Send') {
// balabala
});
}])
.controller('SiblingOneCtrl', ['$scope', function($scope) {
$scope.$on("rootEvent", function(event, 'Data to Send') {
// balabala
});
}])

$rootScope$emit就有些怪异了,按照上面的描述,$rootScope是没有祖先的,所以我们可能会想到其$emit会没有任何作用,但事实并不如此:$rootScope.$emit发布的事件,只能通过$rootScope.$on订阅,而其他$scope对此一无所知:

app.controller('SomeCtrl', ['$rootScope', function($rootScope) {
$rootScope.$emit("rootEvent", 'Data to Send'); // 只有$rootScope可以通过$on订阅从$rootScope.$emit发布的事件
$rootScope.$on("rootEvent", function(event, 'Data to Send') {
// balabala
});
}])
// $scope不能够通过$on订阅从$rootScope.$emit发布的事件
.controller('ParentCtrl', ['$scope', function($scope) {
// 不要这样做
$scope.$on("rootEvent", function(event, 'Data to Send') {
// balabala
});
}]);

退订$rootScope上的订阅事件

当使用$rootScope.$on订阅事件时,需要手动退订事件,一般在其所处$scope$destory事件中退订:

app.controller('SomeCtrl', ['$rootScope', '$scope', function($rootScope, $scope) {
var deregister = $rootScope.$on("rootEvent", function(event, 'Data to Send') {
// balabala
}); $scope.$on('$destory', function() {
deregister(); // 退订事件
});
}])

那通过$scope.$on订阅的事件呢?一般不需要手动退订,因为Angular会帮我们退订,但是如果需要自己控制何时退订事件,也可以通过上述方式进行退订。

事件命名的建议

在开发中,对于变量的命名、函数的命名、文件的命名都有一定的规范,既要保证可读性,也需要保证无混淆性。在Angular的事件机制中,因为事件可能会跨函数,甚至可能跨文件,所以对于事件名一定要保证唯一性,所以建议事件名都加上特定的前缀,以便区分。如下几个例子:

$scope.$emit('trash:delete', data);
$scope.$on('trash:delete', function (event, data) {...}); $scope.$broadcast('trash:clear', data);
$scope.$on('trash:clear', function (event, data) {...});

最新文章

  1. strftime 日期时间格式化
  2. ServletContext与网站计数器
  3. 如何修改Linux主机名
  4. Go文件操作
  5. Python中dict的特点、更新dict、遍历dict
  6. URAL 1930 Ivan&#39;s Car(BFS)
  7. 修改sublime 侧边栏 颜色 等
  8. My first essay
  9. [Swift]LeetCode598. 范围求和 II | Range Addition II
  10. 【存储】RAID磁盘阵列选择
  11. 用D3.js画的人物关系demo
  12. 使用IntelliJ IDEA和Eclipse导入Github项目
  13. json字符串 转Java List 简单方法
  14. vue的优点
  15. mysql中InnoDB表为什么要建议用自增列做主键
  16. JAVA设计模式(全部)
  17. 微信小程序项目实战之天气预报
  18. 上传通用化 VHD 并使用它在 Azure 中创建新 VM
  19. Adobe Acrobat Pro 修改背景色
  20. oracle错误一览表

热门文章

  1. iOS学习之观察者模式
  2. 算法练习:寻找最小的k个数
  3. FFMpeg的码率控制
  4. sql 日期格式输出 - 转
  5. Monte Carlo方法简介(转载)
  6. mysql批量替换单字段
  7. Designing for iOS: Graphics &amp; Performance
  8. 推荐一些不错的计算机书籍(php c mysql linux等等)
  9. Android 学习第18课,单元测试
  10. python数据结构