AngularJS控制器之间的通信方式详解


Posted in Javascript onNovember 03, 2016

本文实例讲述了AngularJS控制器之间的通信方式。分享给大家供大家参考,具体如下:

一、利用作用域的继承方式

由于作用域的继承是基于js的原型继承方式,所以这里分为两种情况,当作用域上面的值为基本类型的时候,修改父作用域上面的值会影响到子作用域,反之,修改子作用域只会影响子作用域的值,不会影响父作用域上面的值;如果需要父作用域与子作用域共享一个值的话,就需要用到后面一种,即作用域上的值为对象,任何一方的修改都能影响另一方,这是因为在js中对象都是引用类型。

基本类型

function Sandcrawler($scope) {
  $scope.location = "Mos Eisley North";
  $scope.move = function(newLocation) {
    $scope.location = newLocation;
  }
}
function Droid($scope) {
  $scope.sell = function(newLocation) {
    $scope.location = newLocation;
  }
}

html:

<div ng-controller="Sandcrawler">
  <p>Location: {{location}}</p>
  <button ng-click="move('Mos Eisley South')">Move</button>
  <div ng-controller="Droid">
    <p>Location: {{location}}</p>
    <button ng-click="sell('Owen Farm')">Sell</button>
  </div>
</div>

对象

function Sandcrawler($scope) {
  $scope.obj = {location:"Mos Eisley North"};
}
function Droid($scope) {
  $scope.summon = function(newLocation) {
    $scope.obj.location = newLocation;
  }
}

html:

<div ng-controller="Sandcrawler">
  <p>Sandcrawler Location: {{location}}</p>
  <div ng-controller="Droid">
    <button ng-click="summon('Owen Farm')">
      Summon Sandcrawler
    </button>
  </div>
</div>

二、基于事件的方式

在一般情况下基于继承的方式已经足够满足大部分情况了,但是这种方式没有实现兄弟控制器之间的通信方式,所以引出了事件的方式。基于事件的方式中我们可以里面作用的$on,$emit,$boardcast这几个方式来实现,其中$on表示事件监听,$emit表示向父级以上的作用域触发事件, $boardcast表示向子级以下的作用域广播事件。参照以下代码:

向上传播事件

function Sandcrawler($scope) {
  $scope.location = "Mos Eisley North";
  $scope.$on('summon', function(e, newLocation) {
    $scope.location = newLocation;
  });
}
function Droid($scope) {
  $scope.location = "Owen Farm";
  $scope.summon = function() {
    $scope.$emit('summon', $scope.location);
  }
}

html:

<div ng-controller="Sandcrawler">
  <p>Sandcrawler Location: {{location}}</p>
  <div ng-controller="Droid">
    <p>Droid Location: {{location}}</p>
    <button ng-click="summon()">Summon Sandcrawler</button>
  </div>
</div>

向下广播事件

function Sandcrawler($scope) {
  $scope.location = "Mos Eisley North";
  $scope.recall = function() {
    $scope.$broadcast('recall', $scope.location);
  }
}
function Droid($scope) {
  $scope.location = "Owen Farm";
  $scope.$on('recall', function(e, newLocation) {
    $scope.location = newLocation;
  });
}

html:

<div ng-controller="Sandcrawler">
  <p>Sandcrawler Location: {{location}}</p>
  <button ng-click="recall()">Recall Droids</button>
  <div ng-controller="Droid">
    <p>Droid Location: {{location}}</p>
  </div>
</div>

从这个用法我们可以引申出一种用于兄弟控制间进行通信的方法,首先我们一个兄弟控制中向父作用域触发一个事件,然后在父作用域中监听事件,再广播给子作用域,这样通过事件携带的参数,实现了数据经过父作用域,在兄弟作用域之间传播。这里要注意的是,通过父元素作为中介进行传递的话,兄弟元素用的事件名不能一样,否则会进入死循环。请看代码:

兄弟作用域之间传播

function Sandcrawler($scope) {
  $scope.$on('requestDroidRecall', function(e) {
    $scope.$broadcast('executeDroidRecall');
  });
}
function Droid($scope) {
  $scope.location = "Owen Farm";
  $scope.recallAllDroids = function() {
    $scope.$emit('requestDroidRecall');
  }
  $scope.$on('executeDroidRecall', function() { 
    $scope.location = "Sandcrawler"
  });
}

html:

<div ng-controller="Sandcrawler">
  <div ng-controller="Droid">
    <h2>R2-D2</h2>
    <p>Droid Location: {{location}}</p>
    <button ng-click="recallAddDroids()">Recall All Droids</button>
  </div>
  <div ng-controller="Droid">
    <h2>C-3PO</h2>
    <p>Droid Location: {{status}}</p>
    <button ng-click="recallAddDroids()">Recall All Droids</button>
  </div>
</div>

三、angular服务的方式

在ng中服务是一个单例,所以在服务中生成一个对象,该对象就可以利用依赖注入的方式在所有的控制器中共享。参照以下例子,在一个控制器修改了服务对象的值,在另一个控制器中获取到修改后的值:

var app = angular.module('myApp', []);
app.factory('instance', function(){
  return {};
});
app.controller('MainCtrl', function($scope, instance) {
 $scope.change = function() {
    instance.name = $scope.test;
 };
});
app.controller('sideCtrl', function($scope, instance) {
  $scope.add = function() {
    $scope.name = instance.name;
  };
});

html:

<div ng-controller="MainCtrl">
   <input type="text" ng-model="test" />
   <div ng-click="change()">click me</div>
</div>
<div ng-controller="sideCtrl">
  <div ng-click="add()">my name {{name}}</div>
</div>

希望本文所述对大家AngularJS程序设计有所帮助。

Javascript 相关文章推荐
javascript 学习笔记(一)DOM基本操作
Apr 08 Javascript
Javascript 闭包引起的IE内存泄露分析
May 23 Javascript
JavaScript修改浏览器tab标题小技巧
Jan 06 Javascript
jQuery动画与特效详解
Feb 01 Javascript
jQuery过滤选择器用法分析
Feb 10 Javascript
jQuery实现的在线答题功能
Apr 12 Javascript
JavaScript中的parse()方法使用简介
Jun 12 Javascript
Angularjs 动态改变title标题(兼容ios)
Dec 29 Javascript
详解基于node的前端项目编译时内存溢出问题
Aug 01 Javascript
Vue封装的可编辑表格插件方法
Aug 28 Javascript
axios使用拦截器统一处理所有的http请求的方法
Nov 02 Javascript
vue用elementui写form表单时,在label里添加空格操作
Aug 13 Javascript
最细致的vue.js基础语法 值得收藏!
Nov 03 #Javascript
AngularJS创建自定义指令的方法详解
Nov 03 #Javascript
JavaScript遍历Json串浏览器输出的结果不统一问题
Nov 03 #Javascript
3种不同的ContextMenu右键菜单实现代码
Nov 03 #Javascript
利用纯Vue.js构建Bootstrap组件
Nov 03 #Javascript
jQuery下拉菜单的实现代码
Nov 03 #Javascript
AngularJS辅助库browserTrigger用法示例
Nov 03 #Javascript
You might like
PHP同时连接多个mysql数据库示例代码
2014/03/17 PHP
yii2 modal弹窗之ActiveForm ajax表单异步验证
2016/06/13 PHP
php 解决substr()截取中文字符乱码问题
2016/07/18 PHP
php-msf源码详解
2017/12/25 PHP
农历与西历对照
2006/09/06 Javascript
jQuery 常见操作实现方式和常用函数方法总结
2011/05/06 Javascript
jquery ajax同步异步的执行最终解决方案
2013/04/26 Javascript
json中换行符的处理方法示例介绍
2014/06/10 Javascript
js获取时间并实现字符串和时间戳之间的转换
2015/01/05 Javascript
JavaScript使用function定义对象并调用的方法
2015/03/23 Javascript
javascript实现图片跟随鼠标移动效果的方法
2015/05/13 Javascript
jQuery数据检索中根据关键字快速定位GridView指定行的实现方法
2016/06/08 Javascript
AngularJS中的DOM操作用法分析
2016/11/04 Javascript
深入学习Bootstrap表单
2016/12/13 Javascript
Web前端框架bootstrap实战【第一次接触使用】
2016/12/28 Javascript
ComboBox(下拉列表框)通过url加载调用远程数据的方法
2017/08/06 Javascript
vue中如何让子组件修改父组件数据
2018/06/14 Javascript
jQuery实现鼠标移到某个对象时弹出显示层功能
2018/08/23 jQuery
vue单页缓存存在的问题及解决方案(小结)
2018/09/25 Javascript
[47:04]EG vs RNG 2019国际邀请赛小组赛 BO2 第二场 8.16
2019/08/18 DOTA
Python中文编码那些事
2014/06/25 Python
python中文分词,使用结巴分词对python进行分词(实例讲解)
2017/11/14 Python
《与孩子一起学编程》python自测题
2018/05/27 Python
Flask框架使用DBUtils模块连接数据库操作示例
2018/07/20 Python
Python中将两个或多个list合成一个list的方法小结
2019/05/12 Python
python 实现围棋游戏(纯tkinter gui)
2020/11/13 Python
CSS3 flex布局之快速实现BorderLayout布局
2015/12/03 HTML / CSS
印尼太阳百货公司网站:Matahari
2018/02/04 全球购物
美国医疗用品、医疗设备和家庭保健用品商店:Medical Supply Depot
2018/07/08 全球购物
在Java开发中如何选择使用哪种集合类
2016/08/09 面试题
实习护理工作自我评价
2013/09/25 职场文书
金融管理毕业生求职信
2014/03/03 职场文书
小学班主任经验交流材料
2014/12/16 职场文书
创业计划书之香辣虾火锅
2019/09/23 职场文书
MySQL8.0.24版本Release Note的一些改进点
2021/04/22 MySQL
2022年显卡天梯图(6月更新)
2022/06/17 数码科技