我在 rails 3.2.6 中有两个类:
- 类 Foo in models/foo.rb (默认在表 foos 中)
- class Bar::Foo in models/bar/foo.rb (将
self.table_name
设置为bar_foos
)
当我进入控制台时,我得到:
> Bar::Foo
=> Bar::Foo(id: ...)
> Foo # or ::Foo
LoadError: expected models/bar/foo.rb to define Foo
怎么了?
我们在 IRC 中解决了这个问题,但核心问题是有一个包含models/**
作为负载路径的config.autoload_paths
glob 集。
Rails的自动加载器迭代加载路径,并固定常量名称。找到存在的文件后,它会尝试加载该文件,然后在常量不可用时引发异常。
所以,发生的事情是Rails有一个负载路径列表,例如:
/models/bar/
/models/
它正在迭代路径,并在 /models/bar/foo.rb
处找到匹配项,然后加载该匹配项(这使Bar::Foo
可用,但不可用Foo
),然后抛出异常,因为Foo
不可用。
在这种情况下,解决方案是删除autoload_paths
设置,以便 Rails 不会找到要为根级常量加载的错误文件。
事实证明,config/applications.rb 中的这一行是问题所在:
config.autoload_paths += Dir[Rails.root.join('app', 'models', '{**}')]
通过显式设置的自动加载,Rails感到困惑;它没有通过命名空间适当地查看模型/,而是查看了它拥有的第一个自动加载文件(错误地,是models/bar/foo.rb),并发现(true)它未能定义Foo(它定义了Bar::Foo)。
所以很明显,Rails 3已经知道在模型/子目录中查找命名空间模型。
感谢 freenode #RubyOnRails 上的 Antiarc 帮助解决这个问题。