AngularJS中控制器之间通信方法

如何在作用域之间通信呢?
    1.创建一个单例服务,然后通过这个服务处理所有子作用域的通信。
    2.通过作用域中的事件处理通信。但是这种方法有一些限制;例如,你并不能广泛的将事件传播到所有监控的作用域中。你必须选择是否与父级作用域或者子作用域通信。


$on、$emit和$broadcast使得event、data在controller之间的传递变的简单。
$emit:子传父  传递event与data
$broadcast:父传子 child controller传递event与data
$on:监听或接收数据。。用于接收event与data

$broadcast、$emit事件必须依靠其他事件(ng-click等)进行触发,而不能单纯写一个这个。。
$on倒是可以直接写,因为它属于监听和接收数据的。


$on的方法中的event事件参数:
    event.name 事件名称
    event.targetScope  发出或者传播原始事件的作用域
    event.currentScope 目前正在处理的事件的作用域
    event.stopPropagation()    一个防止事件进一步传播(冒泡/捕获)的函数(这只适用于使用`$emit`发出的事件)
    event.preventDefault() 这个方法实际上不会做什么事,但是会设置`defaultPrevented`为true。直到事件监听器的实现者采取行动之前它才会检查`defaultPrevented`的值。
    event.defaultPrevented 如果调用了`preventDefault`则为true
 
 
 
 

基于event传播的方式

基于scope继承的方式只能处理父子级控制器之间的通信问题,不能处理兄弟/相邻控制器之间的通信问题。这时候,我们需要使用基于event传播的方式来进行通信,这里,ng为我们提供了三个方法:$on , $emit ,$broadcast ,需要明确的是:这种方法不仅可以处理兄弟scope间的通信问题,对于解决父子scope间的通信也是毫无压力。

子-->父:$emit

整个过程是这样的:

  • 子scope中的控制器通过 $scope.$emit 触发一个事件向上传播

  • 这个事件会经过每一层的父scope,至于处不处理是父scope自己的事情了

  • 如果处理,就在想要处理的那个祖先scope中放一个 $scope.$on 监听着就行了三四三

  • 复制代码
     1 // 父scope上的控制器
     2 function Sandcrawler($scope) {
     3     $scope.location = 'Mos Eisley North';
     4     $scope.$on('summon', function(e, newLocation) {
     5         $scope.location = newLocation;
     6     });
     7 }
     8 // 子scope上的控制器
     9 function Droid($scope) {
    10     $scope.location = 'Owen Farm';
    11     $scope.summon = function() {
    12         $scope.$emit('summon', $scope.location);
    13     }
    14 }
    15 // html
    16 <div ng-controller="Sandcrawler">
    17     <p>Sandcrawler Location: </p>
    18     <div ng-controller="Droid">
    19         <p>Droid Location: </p>
    20         <button ng-click="summon()">Summon Sandcrawler</button>
    21     </div>
    22 </div>
    复制代码

    如果你不想让你的事件再往更上层传播,在 $on 中的处理函数调用e.stopPropagation() 即可。

父-->子:$broadcast

从父到子,用另外一个方法就是了,同样用 $on 监听着,all done,看下面代码:

复制代码
// 父scope上的控制器
function Sandcrawler($scope) {
    $scope.location = 'Mos Eisley North';
    $scope.recall = function() {
        $scope.$broadcast('recall', $scope.location);
    }
}
// 子scope上的控制器
function Droid($scope) {
    $scope.location = 'Owen Farm';
    $scope.$on('recall', function(e, newLocation) {
        $scope.location = newLocation;
    });
}
// html
<div ng-controller="Sandcrawler">
    <p>Sandcrawler Location: </p>
    <button ng-click="recall()">Recall Droids</button>
    <div ng-controller="Droid">
        <p>Droid Location: </p>
    </div>
</div>
复制代码

同级之间

拥有同个父scope的子级scope之间,也就是兄弟/相邻scope之间的通信,其实是借助“奶爸”传递消息的:

  • 子级scope中有谁想传消息了,$emit 一个给“奶爸”

  • 然后通过“奶爸” $broadcast 给所有孩子这个相同的信息,当然发出信息的那个可以选择是否要忽略掉自己发出的信息

复制代码
 1 // 父scope上的控制器
 2 function Sandcrawler($scope) {
 3     $scope.$on('requestDroidRecall', function(e) {
 4         $scope.$broadcast('executeDroidRecall');
 5     });
 6 }
 7 // 子scope上的控制器
 8 function Droid($scope) {
 9     $scope.location = 'Owen Farm';
10     $scope.recallAllDroids = function() {
11         $scope.$emit('requestDroidRecall');
12     }
13     $scope.$on('executeDroidRecall', function() { 
14         $scope.location = 'Sandcrawler';
15     });
16 }
17 // html
18 <div ng-controller="Sandcrawler">
19     <div ng-controller="Droid">
20         <h2>R2-D2</h2>
21         <p>Droid Location: </p>
22         <button ng-click="recallAddDroids()">Recall All Droids</button>
23     </div>
24     <div ng-controller="Droid">
25         <h2>C-3PO</h2>
26         <p>Droid Location: </p>
27         <button ng-click="recallAddDroids()">Recall All Droids</button>
28     </div>
29 </div>
复制代码

上面代码中要注意的是:子scope通过 $emit 发出的事件名不能与父scope用 $broadcast 的事件名一样,如果有传参数,那当然参数可以一样,因为参数就是我们要传的数据。事件名不能一样是为了防止进入死循环。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值