Angular.js中控制器之间的传值详解


Posted in Javascript onApril 24, 2017

前言

每个controller都会有自己的scope,所有的scope都是属于 $rootScope的子或者子的子...

那么问题就好解决了,通过 $rootScope.$broadcast 广播的事件每个controller都能收到事件

另外,我的经验是,尽量不要用event传数据。应该建立一个service来保存数据,并且设置相应getter/setter,具体如下:

 每个controller依赖service, call service.setter(...)

统一的service.setter(...)在改完数据后可以$emit('data-updated')

每个controller里$on('data-updated', function(){ $scope.data = service.getData() })

通过Angular的话可以通过下面四种方法

1、event

这里可以有两种方式,一种是$scope.$emit,然后通过监听$rootScope的事件获取参数;另一种是$rootScope.$broadcast,通过监听$scope的事件获取参数。

这两种方法在最新版本的Angular中已经没有性能区别了,主要就是事件发送的方向不同,可以按实际情况选择。

2、service

可以创建一个专用的事件Service,也可以按照业务逻辑切分,将数据存储在相应的Service中,因为已经有人提过了就不赘述了。

3、$rootScope

这个方法可能会比较dirty一点,胜在方便,也就是把数据存在$rootScope中,这样各个子$scope都可以调用,不过需要注意一下生命周期

4、直接使用$scope.$nextSibling及类似的属性

类似的还有$scope.$parent。这个方法的缺点就更多了,官方不推荐使用任何$开头的属性,既增加了耦合,又需要处理异步的问题,而且scope的顺序也不是固定的。不推荐

另外就是通过本地存储、全局变量或者现代浏览器的postMessage来传递参数了,除非特殊情况,请避免这类方式。

直接建一个service,不要用什么事件,项目一大N多个controller或者N久再去维护会玩死你,service里提供存和取的方法,简单明了,API容易查找和修改,调试也方便无混乱的依赖关系

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

由于作用域的继承是基于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>

2、基于事件的方式

在一般情况下基于继承的方式已经足够满足大部分情况了,但是这种方式没有实现兄弟控制器之间的通信方式,所以引出了事件的方式 。基于事件的方式中我们可以里面作用的$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>

3、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>

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者使用Angular.js能带来一定的帮助,如果有疑问大家可以留言交流,谢谢大家对三水点靠木的支持。

Javascript 相关文章推荐
Date对象格式化函数代码
Jul 17 Javascript
extjs3 combobox取value和text案例详解
Feb 06 Javascript
使用Xcache缓存器加速PHP网站的配置方法
Apr 22 Javascript
Javascript中click与blur事件的顺序详析
Apr 25 Javascript
JavaScript代码执行的先后顺序问题
Oct 29 Javascript
微信小程序图片选择区域裁剪实现方法
Dec 02 Javascript
js实现以最简单的方式将数组元素添加到对象中的方法
Dec 20 Javascript
Bootstrap-table自定义可编辑每页显示记录数
Sep 07 Javascript
小程序怎样让wx.navigateBack更好用的方法实现
Nov 01 Javascript
js实现录音上传功能
Nov 22 Javascript
javascript实现贪吃蛇经典游戏
Apr 10 Javascript
JavaScript逻辑运算符相关总结
Sep 04 Javascript
jquery submit()不能提交表单的解决方法
Apr 24 #jQuery
jquery实现异步加载图片(懒加载图片一种方式)
Apr 24 #jQuery
js前端实现图片懒加载(lazyload)的两种方式
Apr 24 #Javascript
angularjs中的$eval方法详解
Apr 24 #Javascript
浅谈angularjs依赖服务注入写法的注意点
Apr 24 #Javascript
angularjs中回车键触发某一事件的方法
Apr 24 #Javascript
浅谈angularjs中响应回车事件
Apr 24 #Javascript
You might like
php 代码优化之经典示例
2011/03/24 PHP
浅析php插件 HTMLPurifier HTML解析器
2013/07/01 PHP
jQuery+PHP+ajax实现微博加载更多内容列表功能
2014/06/27 PHP
10款实用的PHP开源工具
2015/10/23 PHP
PHP如何获取当前主机、域名、网址、路径、端口等参数
2017/06/09 PHP
基于jQuery的为attr添加id title等效果的实现代码
2011/04/20 Javascript
js两行代码按指定格式输出日期时间
2011/10/21 Javascript
js操作iframe兼容各种主流浏览器示例代码
2013/07/22 Javascript
JS网页播放声音实现代码兼容各种浏览器
2013/09/22 Javascript
原生js的弹出层且其内的窗口居中
2014/05/14 Javascript
window.location.reload 刷新使用分析(去对话框)
2015/11/11 Javascript
使用JavaScript为Kindeditor自定义按钮增加Audio标签
2016/03/18 Javascript
基于JavaScript实现无限加载瀑布流
2017/07/21 Javascript
node.js微信小程序配置消息推送的实现
2019/02/13 Javascript
[52:12]FNATIC vs Infamous 2019国际邀请赛小组赛 BO2 第一场 8.16
2019/08/19 DOTA
Python设计模式之抽象工厂模式
2016/08/25 Python
python、java等哪一门编程语言适合人工智能?
2017/11/13 Python
Python使用Turtle模块绘制五星红旗代码示例
2017/12/11 Python
python多进程使用及线程池的使用方法代码详解
2018/10/24 Python
Python代码实现删除一个list里面重复元素的方法
2019/04/02 Python
Python 学习教程之networkx
2019/04/15 Python
python 申请内存空间,用于创建多维数组的实例
2019/12/02 Python
Python 脚本的三种执行方式小结
2019/12/21 Python
详解python定时简单爬取网页新闻存入数据库并发送邮件
2020/11/27 Python
canvas如何实现多张图片编辑的图片编辑器
2020/03/10 HTML / CSS
日本最佳原创设计品牌:Felissimo(芬理希梦)
2019/03/19 全球购物
办公室内勤岗位职责范本
2013/12/09 职场文书
公交公司毕业生求职信
2014/02/15 职场文书
市级绿色学校申报材料
2014/08/25 职场文书
党的群众路线教育实践活动个人对照检查剖析材料
2014/09/23 职场文书
党员个人总结范文
2015/02/14 职场文书
幼儿园新生开学寄语
2015/05/27 职场文书
风雨哈佛路观后感
2015/06/03 职场文书
工程主管竞聘书
2015/09/15 职场文书
2016年6月份红领巾广播稿
2015/12/21 职场文书
全新239军机修复记
2022/04/05 无线电