Rails Eager Loading和where子句
我渴望用它的关联加载一个模型对象:
user= User.includes(:posts).find(1)
但是在代码中的某些点我想做这样的事情:
user.posts.where(:topic => "x")
但这只是重新运行查询。 所以我想我会这样做:
user.posts.select{|post| post.topic == "x" }
这不会重新运行查询。 但我有几个问题。
首先,这是正确的方法吗?
其次,我对这种情况下的选择有点困惑。 因为当我运行最后一行时,即使我没有使用包含函数,第一次运行查询,然后再运行它,如果我再次运行它,它不会…那么是否涉及某种缓存? 因为当我使用where子句时,它每次都会运行查询。
谢谢。
select
是Enumerable上的Ruby方法。 你第一次跑
user.posts.select{|post| post.topic == "x" }
user
的所有Post
实例:posts
关联将从数据库加载到内存中; 特别是进入ActiveRecord集合对象。 然后在此集合上调用select
,过滤掉集合中的所有Post
实例,其中:topic
属性不是"x"
。
再次运行上面的行不会再次查询数据库,因为您已经将posts
加载到内存中。
当你做下面的includes
user= User.includes(:posts).find(1)
它会急切地为每个返回的User
实例加载:posts
关系(在这种情况下,单个User
,其中:id
为1
)。 现在,您已将所有Post
实例加载到内存中,如上一节中所述。
如果你那么做的事情
user.posts.where(:topic => "x")
你现在告诉rails对Post
运行一个新的查询,这次查找所有Post
实例,其中:topic
是"x"
,其中:user_id
是:id
user
:id
。 where
不像内存中的ARel集合中的“filter”那样工作。
- 如果您想避免另一个错误的查询,
select
是一种过滤内存中结果集的好方法。 - 你可以运行基准测试来找到哪个更快:
- 查询数据库并将另一个ARel集合重新填充到内存中
- 迭代内存中的enumberable并使用Ruby运行filter
-
如果您从不需要所有相关的
:posts
user
:posts
,您可以轻松地将其包含在原始查询中user.includes(:posts).where("posts.topic = ?", 'x')