我正在尝试跨控制器共享数据。用例是一个多步骤表单,在一个输入中输入的数据稍后会在原始控制器之外的多个显示位置使用。下面和 jsfiddle here 中的代码。
HTML
<div ng-controller="FirstCtrl">
<input type="text" ng-model="FirstName"><!-- Input entered here -->
<br>Input is : <strong>{{FirstName}}</strong><!-- Successfully updates here -->
</div>
<hr>
<div ng-controller="SecondCtrl">
Input should also be here: {{FirstName}}<!-- How do I automatically updated it here? -->
</div>
JS
// declare the app with no dependencies
var myApp = angular.module('myApp', []);
// make a factory to share data between controllers
myApp.factory('Data', function(){
// I know this doesn't work, but what will?
var FirstName = '';
return FirstName;
});
// Step 1 Controller
myApp.controller('FirstCtrl', function( $scope, Data ){
});
// Step 2 Controller
myApp.controller('SecondCtrl', function( $scope, Data ){
$scope.FirstName = Data.FirstName;
});
任何帮助是极大的赞赏。
一个简单的解决方案是让您的工厂返回一个对象并让您的控制器使用对同一对象的引用:
JS:
// declare the app with no dependencies
var myApp = angular.module('myApp', []);
// Create the factory that share the Fact
myApp.factory('Fact', function(){
return { Field: '' };
});
// Two controllers sharing an object that has a string in it
myApp.controller('FirstCtrl', function( $scope, Fact ){
$scope.Alpha = Fact;
});
myApp.controller('SecondCtrl', function( $scope, Fact ){
$scope.Beta = Fact;
});
HTML:
<div ng-controller="FirstCtrl">
<input type="text" ng-model="Alpha.Field">
First {{Alpha.Field}}
</div>
<div ng-controller="SecondCtrl">
<input type="text" ng-model="Beta.Field">
Second {{Beta.Field}}
</div>
演示: http://jsfiddle.net/HEdJF/
当应用程序变得更大、更复杂和更难测试时,您可能不想以这种方式从工厂公开整个对象,而是通过 getter 和 setter 提供有限的访问权限:
myApp.factory('Data', function () {
var data = {
FirstName: ''
};
return {
getFirstName: function () {
return data.FirstName;
},
setFirstName: function (firstName) {
data.FirstName = firstName;
}
};
});
使用这种方法,由消费控制器使用新值更新工厂,并观察更改以获取它们:
myApp.controller('FirstCtrl', function ($scope, Data) {
$scope.firstName = '';
$scope.$watch('firstName', function (newValue, oldValue) {
if (newValue !== oldValue) Data.setFirstName(newValue);
});
});
myApp.controller('SecondCtrl', function ($scope, Data) {
$scope.$watch(function () { return Data.getFirstName(); }, function (newValue, oldValue) {
if (newValue !== oldValue) $scope.firstName = newValue;
});
});
HTML:
<div ng-controller="FirstCtrl">
<input type="text" ng-model="firstName">
<br>Input is : <strong>{{firstName}}</strong>
</div>
<hr>
<div ng-controller="SecondCtrl">
Input should also be here: {{firstName}}
</div>
演示: http://jsfiddle.net/27mk1n1o/
我不想为此使用 $watch
。您可以只分配数据,而不是将整个服务分配给控制器的范围。
JS:
var myApp = angular.module('myApp', []);
myApp.factory('MyService', function(){
return {
data: {
firstName: '',
lastName: ''
}
// Other methods or objects can go here
};
});
myApp.controller('FirstCtrl', function($scope, MyService){
$scope.data = MyService.data;
});
myApp.controller('SecondCtrl', function($scope, MyService){
$scope.data = MyService.data;
});
HTML:
<div ng-controller="FirstCtrl">
<input type="text" ng-model="data.firstName">
<br>Input is : <strong>{{data.firstName}}</strong>
</div>
<hr>
<div ng-controller="SecondCtrl">
Input should also be here: {{data.firstName}}
</div>
或者,您可以使用直接方法更新服务数据。
JS:
// A new factory with an update method
myApp.factory('MyService', function(){
return {
data: {
firstName: '',
lastName: ''
},
update: function(first, last) {
// Improve this method as needed
this.data.firstName = first;
this.data.lastName = last;
}
};
});
// Your controller can use the service's update method
myApp.controller('SecondCtrl', function($scope, MyService){
$scope.data = MyService.data;
$scope.updateData = function(first, last) {
MyService.update(first, last);
}
});
您可以通过多种方式在控制器之间共享数据
使用服务 使用 $state.go 服务 使用 stateparams 使用 rootscope
每种方法的说明:
我不打算解释,因为有人使用 $state.go $state.go('book.name', {Name: 'XYZ'}); // 然后从 URL 中获取参数 $state.params.Name; $stateparam 的工作方式与 $state.go 类似,您将其作为对象从发送方控制器传递,并使用 stateparam 使用 $rootscope 收集接收方控制器 (a) 从子控制器向父控制器发送数据 $scope.Save(Obj,function( data) { $scope.$emit('savedata',data); //将数据作为第二个参数传递 }); $scope.$on('savedata',function(event,data) { //接收数据作为第二个参数 }); (b) 从父控制器向子控制器发送数据 $scope.SaveDB(Obj,function(data){ $scope.$broadcast('savedata',data); }); $scope.SaveDB(Obj,function(data){`在此输入代码` $rootScope.$broadcast('saveCallback',data); });
我创建了一个控制路由路径模式之间共享范围的工厂,因此您可以在用户在同一路由父路径中导航时维护共享数据。
.controller('CadastroController', ['$scope', 'RouteSharedScope',
function($scope, routeSharedScope) {
var customerScope = routeSharedScope.scopeFor('/Customer');
//var indexScope = routeSharedScope.scopeFor('/');
}
])
因此,如果用户转到另一个路由路径,例如“/Support”,路径“/Customer”的共享数据将被自动销毁。但是,如果不是这个用户转到“子”路径,例如“/Customer/1”或“/Customer/list”,则范围不会被破坏。
您可以在此处查看示例:http://plnkr.co/edit/OL8of9
$rootScope.$on('$stateChangeSuccess', function(event, toState, toParams, fromState, fromParams){ ... })
有多种方法可以在控制器之间共享数据
角度服务
$broadcast, $emit 方法
父子控制器通信
$根范围
正如我们所知,$rootscope
不是数据传输或通信的首选方式,因为它是一个全局范围,可用于整个应用程序
对于 Angular Js 控制器之间的数据共享,Angular 服务是最佳实践,例如。 .factory
、.service
对于 reference
如果数据从父控制器传输到子控制器,您可以通过 $scope
直接访问子控制器中的父数据如果您使用的是 ui-router
,那么您可以使用 $stateParmas
传递 url 参数,如 id
、{ 5}、key
等
$broadcast
也是在控制器之间从父控制器传输数据到子控制器的好方法,而 $emit
也是从子控制器传输数据到父控制器的好方法
HTML
<div ng-controller="FirstCtrl">
<input type="text" ng-model="FirstName">
<br>Input is : <strong>{{FirstName}}</strong>
</div>
<hr>
<div ng-controller="SecondCtrl">
Input should also be here: {{FirstName}}
</div>
JS
myApp.controller('FirstCtrl', function( $rootScope, Data ){
$rootScope.$broadcast('myData', {'FirstName': 'Peter'})
});
myApp.controller('SecondCtrl', function( $rootScope, Data ){
$rootScope.$on('myData', function(event, data) {
$scope.FirstName = data;
console.log(data); // Check in console how data is coming
});
});
请参阅给定的 link 以了解有关 $broadcast
的更多信息
最简单的解决方案:
我使用了 AngularJS 服务。
第 1 步:我创建了一个名为 SharedDataService 的 AngularJS 服务。
myApp.service('SharedDataService', function () {
var Person = {
name: ''
};
return Person;
});
Step2:创建两个控制器并使用上面创建的服务。
//First Controller
myApp.controller("FirstCtrl", ['$scope', 'SharedDataService',
function ($scope, SharedDataService) {
$scope.Person = SharedDataService;
}]);
//Second Controller
myApp.controller("SecondCtrl", ['$scope', 'SharedDataService',
function ($scope, SharedDataService) {
$scope.Person = SharedDataService;
}]);
Step3:只需在视图中使用创建的控制器。
<body ng-app="myApp">
<div ng-controller="FirstCtrl">
<input type="text" ng-model="Person.name">
<br>Input is : <strong>{{Person.name}}</strong>
</div>
<hr>
<div ng-controller="SecondCtrl">
Input should also be here: {{Person.name}}
</div>
</body>
要查看此问题的有效解决方案,请按以下链接
https://codepen.io/wins/pen/bmoYLr
.html 文件:
<!DOCTYPE html>
<html>
<script src="https://ajax.googleapis.com/ajax/libs/angularjs/1.6.9/angular.min.js"></script>
<body ng-app="myApp">
<div ng-controller="FirstCtrl">
<input type="text" ng-model="Person.name">
<br>Input is : <strong>{{Person.name}}</strong>
</div>
<hr>
<div ng-controller="SecondCtrl">
Input should also be here: {{Person.name}}
</div>
//Script starts from here
<script>
var myApp = angular.module("myApp",[]);
//create SharedDataService
myApp.service('SharedDataService', function () {
var Person = {
name: ''
};
return Person;
});
//First Controller
myApp.controller("FirstCtrl", ['$scope', 'SharedDataService',
function ($scope, SharedDataService) {
$scope.Person = SharedDataService;
}]);
//Second Controller
myApp.controller("SecondCtrl", ['$scope', 'SharedDataService',
function ($scope, SharedDataService) {
$scope.Person = SharedDataService;
}]);
</script>
</body>
</html>
FirstCtrl
是父级并获得 SecondCtrl
(子级)也需要的承诺?我不想进行两次 api 调用。谢谢。
还有另一种不使用 $watch 的方法,使用 angular.copy:
var myApp = angular.module('myApp', []);
myApp.factory('Data', function(){
var service = {
FirstName: '',
setFirstName: function(name) {
// this is the trick to sync the data
// so no need for a $watch function
// call this from anywhere when you need to update FirstName
angular.copy(name, service.FirstName);
}
};
return service;
});
// Step 1 Controller
myApp.controller('FirstCtrl', function( $scope, Data ){
});
// Step 2 Controller
myApp.controller('SecondCtrl', function( $scope, Data ){
$scope.FirstName = Data.FirstName;
});
有多种方法可以做到这一点。
事件 - 已经很好地解释了。 ui 路由器 - 上面解释过。服务 - 更新方法显示在 BAD 上方 - 观察变化。另一个父子方法而不是发射和广播 -
*
<superhero flight speed strength> Superman is here! </superhero>
<superhero speed> Flash is here! </superhero>
*
app.directive('superhero', function(){
return {
restrict: 'E',
scope:{}, // IMPORTANT - to make the scope isolated else we will pollute it in case of a multiple components.
controller: function($scope){
$scope.abilities = [];
this.addStrength = function(){
$scope.abilities.push("strength");
}
this.addSpeed = function(){
$scope.abilities.push("speed");
}
this.addFlight = function(){
$scope.abilities.push("flight");
}
},
link: function(scope, element, attrs){
element.addClass('button');
element.on('mouseenter', function(){
console.log(scope.abilities);
})
}
}
});
app.directive('strength', function(){
return{
require:'superhero',
link: function(scope, element, attrs, superHeroCtrl){
superHeroCtrl.addStrength();
}
}
});
app.directive('speed', function(){
return{
require:'superhero',
link: function(scope, element, attrs, superHeroCtrl){
superHeroCtrl.addSpeed();
}
}
});
app.directive('flight', function(){
return{
require:'superhero',
link: function(scope, element, attrs, superHeroCtrl){
superHeroCtrl.addFlight();
}
}
});
不知道我在哪里选择了这种模式,但是为了在控制器之间共享数据并减少 $rootScope 和 $scope 这很有效。这让人想起拥有发布者和订阅者的数据复制。希望能帮助到你。
服务:
(function(app) {
"use strict";
app.factory("sharedDataEventHub", sharedDataEventHub);
sharedDataEventHub.$inject = ["$rootScope"];
function sharedDataEventHub($rootScope) {
var DATA_CHANGE = "DATA_CHANGE_EVENT";
var service = {
changeData: changeData,
onChangeData: onChangeData
};
return service;
function changeData(obj) {
$rootScope.$broadcast(DATA_CHANGE, obj);
}
function onChangeData($scope, handler) {
$scope.$on(DATA_CHANGE, function(event, obj) {
handler(obj);
});
}
}
}(app));
获取新数据的控制器,即发布者会做这样的事情..
var someData = yourDataService.getSomeData();
sharedDataEventHub.changeData(someData);
也在使用这个新数据的控制器,称为订阅者,会做这样的事情......
sharedDataEventHub.onChangeData($scope, function(data) {
vm.localData.Property1 = data.Property1;
vm.localData.Property2 = data.Property2;
});
这适用于任何情况。因此,当主控制器初始化并获取数据时,它将调用 changeData 方法,然后将其广播给该数据的所有订阅者。这减少了我们的控制器之间的耦合。
sharedDataEventHub.changeData(newData)
sharedDataEventHub.changeData(newData)
,那么任何依赖该数据的子或控制器将在其控制器主体中的某个位置具有以下内容:{ 2} 如果你使用 UI-Router 这可以从状态配置中注入。
正如@Mann 在接受答案的评论之一中指出的那样,如果刷新页面,该解决方案将不起作用。
解决方案是使用 localStorage 或 sessionStorage 来临时持久化要跨控制器共享的数据。
您可以创建一个 sessionService 的 GET 和 SET 方法,加密和解密数据并从 localStorage 或 sessionStorage 读取数据。因此,现在您可以直接使用此服务通过您想要的任何控制器或服务来读取和写入存储中的数据。这是一种开放的方法,而且很简单,否则您创建一个 DataSharing Service 并在其中使用 localStorage - 这样,如果刷新页面,该服务将尝试检查存储并通过您在其中公开或私有的 Getter 和 Setter 回复此服务文件。
做简单(用 v1.3.15 测试):
<article ng-controller="ctrl1 as c1">
<label>Change name here:</label>
<input ng-model="c1.sData.name" />
<h1>Control 1: {{c1.sData.name}}, {{c1.sData.age}}</h1>
</article>
<article ng-controller="ctrl2 as c2">
<label>Change age here:</label>
<input ng-model="c2.sData.age" />
<h1>Control 2: {{c2.sData.name}}, {{c2.sData.age}}</h1>
</article>
<script>
var app = angular.module("MyApp", []);
var dummy = {name: "Joe", age: 25};
app.controller("ctrl1", function () {
this.sData = dummy;
});
app.controller("ctrl2", function () {
this.sData = dummy;
});
</script>
dummy
对两个控制器都是全局的,而不是通过使用 $scope 或 controllerAs 继承或使用服务以更模块化的方式共享。
return { FirstName: '' };
这样的对象或返回包含与对象交互的方法的对象,不如返回类的实例 (function
),这样当您使用对象时,它具有特定的类型,它不仅仅是一个“Object{}”?newValue !== oldValue
检查,因为如果 oldValue 和 newValue 相同,Angular 不会执行该函数。唯一的例外是如果您关心初始值。请参阅:docs.angularjs.org/api/ng/type/$rootScope.Scope#$watch