草庐IT

javascript - 如何将设置对象从 Controller 传递给服务?

coder 2024-07-28 原文

tl;博士;

我需要传达几个服务需要的状态,并且源自绑定(bind)到 Controller 范围的数据。什么是好的和“Angular 禅宗”方式来做到这一点?

背景故事

我正在开发单页应用程序,经过深思熟虑后决定使用 AngularJS。页面的布局方式类似于:

实际的布局并不重要,对于类似的布局,概念保持不变。 我需要将绑定(bind)到 SettingsController 范围的信息传达给 ngView 中的 Controller 所需的服务。当用户对任何 slider 进行修改时,我还需要更新从 Controller 中的服务获取的内容。

我尝试过的

我想到的唯一方法是:http://jsfiddle.net/5sNcG/我必须自己编写一个绑定(bind)并将监听器添加到范围更改中。我可能离这里很远,并且有一种明显的“Angular ”方式可以做到这一点 - 但是尽管我付出了努力,我还是找不到它。

/code from fiddle.
var app = angular.module("myApp",[]);

app.controller("HomeCtrl",function($scope,FooService,$interval){
    FooService.change(function(){
        console.log("HI",FooService.getFoo());
        $scope.foo = FooService.getFoo();
    });
});

app.factory("Configuration",function(){
    var config = {data:'lol'};
    var callbacks = [];
    return {
        list:function(){ return config;},
        update:function(){
            callbacks.forEach(function(x){ x();});
        },
        change:function(fn){
            callbacks.push(fn); // I never remove these, so this is a memory leak!
        }
    }
});
app.service("FooService",function(Configuration){
    return {
        getFoo: function(){
            return Configuration.list().data+" bar";    
        },change:function(fn){
            Configuration.change(fn);    
        }
    }
});
app.controller("SettingsCtrl",function($scope,Configuration){

    $scope.config = Configuration.list();
    $scope.$watch('config',function(){
        Configuration.update();
    },true);
});

我也考虑过 $rootScope 广播,但这看起来更像全局状态

无论我尝试什么,我都有一个具有全局状态的单例,我们都知道 I don't want a singleton .

因为这似乎是一个相当常见的 Angular 用例。解决这个问题的惯用方法是什么?

最佳答案

我过去遇到过类似的事情,并考虑了四种可能的方法:

  • 使用$broadcast
  • $rootScope 上存储设置
  • 观察者模式(如您所见)
  • 在 Controller 中使用 $watch

以下是我对每个问题的看法:

$广播

在我看到的 AngularJS 演示中,Miško Hevery 谈到了 $broadcast(即事件)的使用及其用例。要点是 $broadcast 更适用于对与您正在处理的任何事情没有紧密耦合的事件使用react,否则可能更可取。同样关于这个主题,Best Practices Angular wiki 上的指南建议:

Only use .$broadcast(), .$emit() and .$on() for atomic events: Events that are relevant globally across the entire app (such as a user authenticating or the app closing).

在这里,由于您的设置与填充 ng-view 的任何内容密切相关,因此建议使用 $broadcast 的替代方法更可取。

$rootScope

正如您提到的(并希望避免),这是一个全局状态。向我的整个应用程序公开设置不是/不是我的个人偏好,尽管它通常是简单的选择。我个人保留 $rootScope 用于配置设置和“软”变量,如页面标题等。我不会选择使用此选项。

观察者模式

针对配置工厂注册回调是一种可靠的方法。关于持久回调,您可以在作用域上监听 $destroy 事件,调用配置工厂的 remove 方法来删​​除回调。这可以被认为是如何使用 $broadcast 的一个很好的例子; Controller 关注事件并且必须对其使用react,但事件本身并不特定于 Controller /配置服务共享的数据。

$ watch

通过使用共享服务,可以将其注入(inject)到任何与设置相关的 Controller 中。现在,对配置的任何更改都会触发您的回调,而某些 View 可能只与一两个配置设置有关。 $watch 将使您更容易观察仅对那些属性的更改。我不能说开销与注册回调,但这对我来说是最“有 Angular ”的方式。

这是如何使用 $watch 实现的:

var app = angular.module("myApp",[]);

app.factory("Configuration",function(){
   var data = {
     settingOne: true,
     settingTwo: false 
   };
   return data;
})

app.controller("SettingsCtrl",function($scope, Configuration){
  // do something
})

app.controller("HomeCtrl",function($scope, Configuration){
   // detect any change to configuration settings
   $scope.$watch(function() {
     return Configuration;
   }, function(data) {
     // do something
   }, true)

   // alternatively only react to settingTwo changing
   $scope.$watch(function() {
     return Configuration.settingTwo
   }, function(data) {
     // do something
   })
})

请注意,如果您需要稍微复杂一些的配置工厂,您可以转而使用 getter/setter 方法并将配置设置本身保密。然后,在 $watch 中,您应该监视方法调用而不是属性本身。

更新:

在回答时,我更喜欢在 Controller 中使用 $watch 的方法。在使用框架开发了一段时间后,我现在尝试将 $watch 完全置于 Controller 之外,而不是尽可能地在值发生变化时直接调用函数,或者通过利用 ng-change

这样做的一个原因是它增加了测试 Controller 的复杂性,但也许更重要的是它效率低下:对于每个 $digest 循环 Angular 调用,每个注册的 $watch无论如何都会被评估,并且它很可能会响应对现有 $watch 的值所做的更改。

与其从这个 Angular 总结缺点和解决方案,不如在此处有一篇关于此问题的非常好的文章:Angular JS - you probably shouldn't use $watch in your controllers .

关于javascript - 如何将设置对象从 Controller 传递给服务?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/21515106/

有关javascript - 如何将设置对象从 Controller 传递给服务?的更多相关文章

  1. ruby - 如何使用 Nokogiri 的 xpath 和 at_xpath 方法 - 2

    我正在学习如何使用Nokogiri,根据这段代码我遇到了一些问题:require'rubygems'require'mechanize'post_agent=WWW::Mechanize.newpost_page=post_agent.get('http://www.vbulletin.org/forum/showthread.php?t=230708')puts"\nabsolutepathwithtbodygivesnil"putspost_page.parser.xpath('/html/body/div/div/div/div/div/table/tbody/tr/td/div

  2. ruby - 如何从 ruby​​ 中的字符串运行任意对象方法? - 2

    总的来说,我对ruby​​还比较陌生,我正在为我正在创建的对象编写一些rspec测试用例。许多测试用例都非常基础,我只是想确保正确填充和返回值。我想知道是否有办法使用循环结构来执行此操作。不必为我要测试的每个方法都设置一个assertEquals。例如:describeitem,"TestingtheItem"doit"willhaveanullvaluetostart"doitem=Item.new#HereIcoulddotheitem.name.shouldbe_nil#thenIcoulddoitem.category.shouldbe_nilendend但我想要一些方法来使用

  3. ruby - 使用 RubyZip 生成 ZIP 文件时设置压缩级别 - 2

    我有一个Ruby程序,它使用rubyzip压缩XML文件的目录树。gem。我的问题是文件开始变得很重,我想提高压缩级别,因为压缩时间不是问题。我在rubyzipdocumentation中找不到一种为创建的ZIP文件指定压缩级别的方法。有人知道如何更改此设置吗?是否有另一个允许指定压缩级别的Ruby库? 最佳答案 这是我通过查看ruby​​zip内部创建的代码。level=Zlib::BEST_COMPRESSIONZip::ZipOutputStream.open(zip_file)do|zip|Dir.glob("**/*")d

  4. ruby - 使用 ruby​​ 和 savon 的 SOAP 服务 - 2

    我正在尝试使用ruby​​和Savon来使用网络服务。测试服务为http://www.webservicex.net/WS/WSDetails.aspx?WSID=9&CATID=2require'rubygems'require'savon'client=Savon::Client.new"http://www.webservicex.net/stockquote.asmx?WSDL"client.get_quotedo|soap|soap.body={:symbol=>"AAPL"}end返回SOAP异常。检查soap信封,在我看来soap请求没有正确的命名空间。任何人都可以建议我

  5. python - 如何使用 Ruby 或 Python 创建一系列高音调和低音调的蜂鸣声? - 2

    关闭。这个问题是opinion-based.它目前不接受答案。想要改进这个问题?更新问题,以便editingthispost可以用事实和引用来回答它.关闭4年前。Improvethisquestion我想在固定时间创建一系列低音和高音调的哔哔声。例如:在150毫秒时发出高音调的蜂鸣声在151毫秒时发出低音调的蜂鸣声200毫秒时发出低音调的蜂鸣声250毫秒的高音调蜂鸣声有没有办法在Ruby或Python中做到这一点?我真的不在乎输出编码是什么(.wav、.mp3、.ogg等等),但我确实想创建一个输出文件。

  6. ruby-on-rails - 如何验证 update_all 是否实际在 Rails 中更新 - 2

    给定这段代码defcreate@upgrades=User.update_all(["role=?","upgraded"],:id=>params[:upgrade])redirect_toadmin_upgrades_path,:notice=>"Successfullyupgradeduser."end我如何在该操作中实际验证它们是否已保存或未重定向到适当的页面和消息? 最佳答案 在Rails3中,update_all不返回任何有意义的信息,除了已更新的记录数(这可能取决于您的DBMS是否返回该信息)。http://ar.ru

  7. ruby-openid:执行发现时未设置@socket - 2

    我在使用omniauth/openid时遇到了一些麻烦。在尝试进行身份验证时,我在日志中发现了这一点:OpenID::FetchingError:Errorfetchinghttps://www.google.com/accounts/o8/.well-known/host-meta?hd=profiles.google.com%2Fmy_username:undefinedmethod`io'fornil:NilClass重要的是undefinedmethodio'fornil:NilClass来自openid/fetchers.rb,在下面的代码片段中:moduleNetclass

  8. ruby-on-rails - 按天对 Mongoid 对象进行分组 - 2

    在控制台中反复尝试之后,我想到了这种方法,可以按发生日期对类似activerecord的(Mongoid)对象进行分组。我不确定这是完成此任务的最佳方法,但它确实有效。有没有人有更好的建议,或者这是一个很好的方法?#eventsisanarrayofactiverecord-likeobjectsthatincludeatimeattributeevents.map{|event|#converteventsarrayintoanarrayofhasheswiththedayofthemonthandtheevent{:number=>event.time.day,:event=>ev

  9. ruby-on-rails - 'compass watch' 是如何工作的/它是如何与 rails 一起使用的 - 2

    我在我的项目目录中完成了compasscreate.和compassinitrails。几个问题:我已将我的.sass文件放在public/stylesheets中。这是放置它们的正确位置吗?当我运行compasswatch时,它不会自动编译这些.sass文件。我必须手动指定文件:compasswatchpublic/stylesheets/myfile.sass等。如何让它自动运行?文件ie.css、print.css和screen.css已放在stylesheets/compiled。如何在编译后不让它们重新出现的情况下删除它们?我自己编译的.sass文件编译成compiled/t

  10. ruby - 具有身份验证的私有(private) Ruby Gem 服务器 - 2

    我想安装一个带有一些身份验证的私有(private)Rubygem服务器。我希望能够使用公共(public)Ubuntu服务器托管内部gem。我读到了http://docs.rubygems.org/read/chapter/18.但是那个没有身份验证-如我所见。然后我读到了https://github.com/cwninja/geminabox.但是当我使用基本身份验证(他们在他们的Wiki中有)时,它会提示从我的服务器获取源。所以。如何制作带有身份验证的私有(private)Rubygem服务器?这是不可能的吗?谢谢。编辑:Geminabox问题。我尝试“捆绑”以安装新的gem..

随机推荐