gpt4 book ai didi

ruby-on-rails - Rails Eager Loading 和 where 子句

转载 作者:行者123 更新时间:2023-12-04 06:01:52 24 4
gpt4 key购买 nike

我渴望加载一个模型对象及其关联:

user= User.includes(:posts).find(1)

但是在代码中的某些点我想做这样的事情:
user.posts.where(:topic => "x")

但这只是再次重新运行查询。所以相反,我想我会这样做:
user.posts.select{|post| post.topic == "x" }

这不会重新运行查询。但我有几个问题。

首先,这是正确的方法吗?

其次,我对 select 在这种情况下的作用有点困惑。因为当我运行最后一行时,即使我没有使用 include 函数,它第一次运行查询,然后如果我再次运行它,它不会......那么是否涉及某种缓存?
因为当我使用 where 子句时,它每次都会运行查询。

谢谢你。

最佳答案

select 是 Enumerable 上的 Ruby 方法。第一次运行

user.posts.select{|post| post.topic == "x" }

全部 Post user 的实例的 :posts关联将从数据库加载到内存中;具体到 ActiveRecord 集合对象。 select然后在这个集合上调用,过滤掉所有 Post集合中带有 :topic 的实例属性不是 "x" .

第二次运行上面的行不会再次查询数据库,因为你已经加载了 posts进入内存。

当你做一个 includes像下面
user= User.includes(:posts).find(1)

它将急切加载 :posts每个的关系 User返回的实例(在这种情况下,单个 User,其中 :id1)。您现在拥有所有 Post实例加载到内存中,如上一节所述。

如果你然后做类似的事情
user.posts.where(:topic => "x")

你现在告诉 rails 运行 新品 查询 Post ,这次找到所有 Post实例其中 :topic"x"哪里 :user_id:iduser . where不像内存中的 ARel 集合上的“过滤器”那样工作。

  • 如果你想避免另一个糟糕的查询,select是过滤内存结果集的好方法。
  • 您可以运行基准测试来找出哪个更快:
  • 查询数据库并将另一个 ARel 集合重新填充到内存中
  • 迭代内存中的可编号并使用 Ruby 运行过滤器
  • 如果您永远不需要 全部 相关 :postsuser ,您可以轻松地将其包含在原始查询中
    user.includes(:posts).where("posts.topic = ?", 'x')
  • 关于ruby-on-rails - Rails Eager Loading 和 where 子句,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/15523627/

    24 4 0
    Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
    广告合作:1813099741@qq.com 6ren.com