7

我们总是习惯将我们的应用程序配置放入环境文件中。这对生产管理没有好处,所以现在我们通过初始化程序加载它:

# myinitializer.rb
ApplicationConfig = YAML.load_file("#{Rails.root}/config/application/default.yml").symbolize_keys()

一旦我们开始通过 访问配置ApplicationConfig,应用程序测试性能就会变得更糟。一个 rspec 套件从 4 秒下降到 30 秒。

在我们的应用程序控制器中,我们需要使用 a 执行一些操作before_filter,其工作原理如下:

before_filter :extra_control

def extra_control
  if ApplicationConfig.some_flag
    ...
  end
end

声明一个指向ApplicationConfig完全恢复性能的变量:

config = ApplicationConfig

def extra_control
  if config.some_flag
    ...
  end
end

为什么?为什么通过全局变量访问它会破坏性能?我们在视图和其他控制器的整个代码库中执行此操作。我们是否需要以不同的方式执行此操作,例如通过将实例变量注入所有控制器?

编辑:我们确实验证了从 YAML 加载配置的代码以任何一种方式都被调用一次,因此重复加载似乎不是根本原因。

编辑:事实证明,这是一个由作为字符串而不是布尔值加载的设置变量引起的错误,导致应用程序进入测试睡眠模式:(对不起,感谢您的尝试。我生命中的 3 天我永远不会回来了!

4

4 回答 4

0

我无法解释为什么您的方法很慢,但也许值得研究另一种处理设置的方式。我更喜欢拥有一个包含配置信息的单例类的想法:

class SiteSettings
    def initialize
        @settings = {}
    end

    @@instance = SiteSettings.new

    def set_setting(key, value)
        @settings ||= {}
        @settings[key] = value
    end

    def self.setting(key, value)
        @@instance.set_setting(key, value)
    end

    def settings
        @settings
    end

    def self.method_missing(meth, *args, &block)
        if @@instance.settings.has_key?(meth)
            @@instance.settings[meth]
        else
            super
        end
    end

    setting :test_setting, 'test'
    setting :another_test_setting '.....'
end

puts SiteSettings.test_setting
puts SiteSettings.another_test_setting

Discourse 应用程序采用了类似的方法。

于 2013-05-07T17:20:08.500 回答
0

尝试使用 const:

APPLICATIONCONFIG = YAML.load(File.read(File.expand_path("#{RAILS_ROOT}/config/application/default.yml", __FILE__)))

或者

APPLICATIONCONFIG = YAML::load(File.open("#{RAILS_ROOT}/config/application/default.yml"))

我认为它使您的文件变慢,因为加载文件但未打开/读取。

于 2013-05-08T07:53:35.537 回答
0

我认为这将有助于确定 YAML 文件加载\读取是否缓慢,或者 YAML 解析是否缓慢。我的猜测是,每次调用ApplicationConfig变量时,ruby 都会解析 YAML 文件。您可以尝试将其重命名为APPLICATION_CONFIG

APPLICATION_CONFIG = YAML.load(File.read(Rails.root.join('config', 'application', 'my_config.yml')))

请不要在所有控制器中注入实例变量 :)

于 2013-05-08T09:51:16.517 回答
0

这里的其他答案没有什么不同 - 这是由一个不相关的问题引起的。作为记录

  • 初始化器只被 rspec 调用一次
  • 使用 ApplicationConfig 或 APPLICATION_CONFIG 没有区别
  • 分配给变量也不会产生速度差异
于 2013-05-12T21:30:37.687 回答