加入收藏 | 设为首页 | 会员中心 | 我要投稿 李大同 (https://www.lidatong.com.cn/)- 科技、建站、经验、云计算、5G、大数据,站长网!
当前位置: 首页 > 百科 > 正文

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

发布时间:2020-12-17 04:20:24 所属栏目:百科 来源:网络整理
导读:我渴望用它的关联加载一个模型对象: user= User.includes(:posts).find(1) 但是在代码中的某些点我想做这样的事情: user.posts.where(:topic = "x") 但这只是重新运行查询.所以我想我会这样做: user.posts.select{|post| post.topic == "x" } 这不会重新
我渴望用它的关联加载一个模型对象:
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" }

用户的所有Post实例:posts关联将从数据库加载到内存中;特别是进入ActiveRecord集合对象.然后在此集合上调用select,过滤掉集合中的所有Post实例,其中:topic属性不是“x”.

再次运行上面的行不会再次查询数据库,因为您已经将帖子加载到内存中.

当你做下面的包括

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.哪里不像内存中的ARel集合中的“过滤器”那样工作.

>如果您想避免另一个错误的查询,select是一种过滤内存中结果集的好方法.
>你可以运行基准来找到哪个更快:

>查询数据库并将另一个ARel集合重新填充到内存中
>迭代内存中的可枚举并使用Ruby运行过滤器

>如果您从不需要所有相关的:用户帖子,您可以轻松地将其包含在原始查询中

user.includes(:posts).where("posts.topic = ?",'x')

(编辑:李大同)

【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容!

    推荐文章
      热点阅读