我在 rails 3.2.6 中有两个类:
- models/foo.rb 中的类 Foo(默认情况下在表 foos 中)
- 模型/bar/foo.rb 中的类 Bar::Foo (设置
self.table_name
为bar_foos
)
当我进入控制台时,我得到:
> Bar::Foo
=> Bar::Foo(id: ...)
> Foo # or ::Foo
LoadError: expected models/bar/foo.rb to define Foo
怎么了?
我在 rails 3.2.6 中有两个类:
self.table_name
为bar_foos
)当我进入控制台时,我得到:
> Bar::Foo
=> Bar::Foo(id: ...)
> Foo # or ::Foo
LoadError: expected models/bar/foo.rb to define Foo
怎么了?
我们在 IRC 中解决了这个问题,但核心问题是存在一个config.autoload_paths
包含models/**
作为加载路径的 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/ 下,而是查看了它拥有的第一个自动加载文件(错误地是 models/bar/foo.rb)并发现(true)它未能定义 Foo(它定义了 Bar::福)。
所以很明显,Rails 3 已经知道在模型/子目录中查找命名空间模型。
感谢 freenode 上的 Antiarc #RubyOnRails 帮助解决这个问题。