这是我的示例 rails 模型类(在 app/models/user.rb 中):
class User < ActiveRecord::Base
puts "loading user model"
include Auth::User # module is in the Auth engine
end
这是一个名为 auth 的引擎中的模块(文件位于 vendor/engines/app/models/auth/user.rb 中)
module Auth
module User
puts "loading module"
self.included(base)
puts "module is included"
end
end
end
当 rails (3.1.1 ruby 1.92) 启动时(在非开发中),我看到“加载用户模型”和“加载模块”,但没有看到“包含模块”。
如果我将模块的名称和文件名从 models/auth/user.rb 更改为 user_auth.rb,如果模块在 rails init 进程中“足够早”包含,我可能会看到“包含”(见下文)。我想将模块的名称和文件名保留为用户。我认为没有理由必须改变这一点。当我加载 User 模型(以及当 User 模型包含 auth/user 模块时)时,这无关紧要。
注意:当我在上面说“MAY”时,这是我观察到的:如果 rails “碰巧”需要加载 UsersController,那么它将恰好加载 User 对象,并且仅当我重命名用户时才会包含 User 模块模块到 UserAuth(或与模型名称不同的名称)。由于路由,我“碰巧”加载了 UsersController。我知道这一点是因为我打印了一个堆栈跟踪。所以我可以通过 1) 重命名模块和 2) 强制它使用控制器加载来包含模块。
注意:继续...仅当包含它的类在 Rails 启动过程中的某个时间早期加载/需要时才加载该模块。这就是为什么通过强制rails“看到”UsersController(因为加载路由)来强制User模型“提前”加载的原因也将包括auth/user模块。
注意:如果我运行控制台(同样,在非开发模式下)然后执行 require 'user',则不包含该模块。因此,如果我“等待”直到 rails 完全加载后,当我需要用户时,auth/user 模块不包括在内。我认为这有点不准确,我看到在 rails init 过程中加载了用户模型,我看到正在加载 auth/user 模块,但当时没有看到 auth/user 模块。如果用户模型在 rails init 进程中“早期”加载,我只会看到包含的模块。或者它可能“在 rails init 进程中的某个神奇时间加载)。
任何人都可以解释一下吗?我尝试了许多不同的方法来包含模块,例如打开 User 类并重新包含模块。这似乎有时有效,但并非始终如一。