有没有办法急于加载多态关联的关联?

时间:2011-12-28 05:08:25

标签: ruby-on-rails polymorphic-associations eager-loading

艺术家有很多活动(基本上是用户之间互动的缓存):

class Activity < ActiveRecord::Base

  belongs_to :receiver, :class_name => 'Artist', :foreign_key => :receiver_id #owns the stuff done "TO" him
  belongs_to :link, :polymorphic => true
  belongs_to :creator, :class_name => 'Artist', :foreign_key => :creator_id #person who initiated the activity

end

例如:

  Activity.create(:receiver_id => author_id, :creator_id => artist_id, :link_id => id, :link_type => 'ArtRating') 

我想为每位艺术家创建一个活动流页面,其中包含不同类型的活动列表,ArtRatings(喜欢,不喜欢),收藏,关注等。

控制器如下所示:

class ActivityStreamController < ApplicationController
  def index
    @activities = @artist.activities.includes([:link,:creator,:receiver]).order("id DESC").limit(30)
  end
end

db调用正确地急切地加载多态链接对象:

  SELECT "activities".* FROM "activities" WHERE (("activities"."receiver_id" = 6 OR "activities"."creator_id" = 6)) ORDER BY id DESC LIMIT 30
  ArtRating Load (0.5ms)  SELECT "art_ratings".* FROM "art_ratings" WHERE "art_ratings"."id" IN (137, 136, 133, 130, 126, 125, 114, 104, 103, 95, 85, 80, 73, 64)
  SELECT "follows".* FROM "follows" WHERE "follows"."id" IN (14, 10)
  SELECT "favorites".* FROM "favorites" WHERE "favorites"."id" IN (25, 16, 14)

但是当我显示每个ArtRating时,我还需要引用属于帖子的帖子标题。在视图中,如果我这样做:

activity.link.post

它为每个art_rating的帖子执行单独的数据库调用。有没有办法急切加载帖子?

更新问题:

如果无法使用'includes'语法实现对帖子的热切加载,是否有办法自行手动执行热切加载查询并将其注入@activities对象?

我在DB日志中看到:

SELECT "art_ratings".* FROM "art_ratings" WHERE "art_ratings"."id" IN (137, 136, 133, 130, 126, 125, 114, 104, 103, 95, 85, 80, 73, 64)

有没有办法可以从@activities对象访问这个id列表?如果是这样,我可以执行2个额外的查询,1可以获取该列表中的art_ratings.post_id(s),另一个用于选择post_id列表中的所有帖子。然后以某种方式将'post'结果注入@activities,以便当我遍历集合时它可以作为activity.link.post使用。可能的?

5 个答案:

答案 0 :(得分:4)

TL; DR我的解决方案让artist.created_activities.includes(:link)急切加载您想要的所有内容

这是我的第一次尝试:https://github.com/JohnAmican/music

一些注意事项:

  • 我依赖default_scope,所以这不是最佳的。
  • 看起来你正在使用STI。我的解决方案没有。这意味着你不能简单地给艺术家打电话activities;您必须引用created_activitiesreceived_activities。可能有办法解决这个问题。如果我找到任何东西,我会更新。
  • 我改变了一些名字,因为这对我来说很困惑。

如果你进入控制台并执行created_activities.includes(:link),那么适当的东西会被加载:

irb(main):018:0> artist.created_activities.includes(:link)
  Activity Load (0.2ms)  SELECT "activities".* FROM "activities" WHERE "activities"."creator_id" = ?  [["creator_id", 1]]
  Rating Load (0.3ms)  SELECT "ratings".* FROM "ratings" WHERE "ratings"."id" IN (1)
  RatingExplanation Load (0.3ms)  SELECT "rating_explanations".* FROM "rating_explanations" WHERE "rating_explanations"."rating_id" IN (1)
  Following Load (0.3ms)  SELECT "followings".* FROM "followings" WHERE "followings"."id" IN (1)
  Favorite Load (0.2ms)  SELECT "favorites".* FROM "favorites" WHERE "favorites"."id" IN (1)
=> #<ActiveRecord::Relation [#<Activity id: 1, receiver_id: 2, creator_id: 1, link_id: 1, link_type: "Rating", created_at: "2013-10-31 02:36:27", updated_at: "2013-10-31 02:36:27">, #<Activity id: 2, receiver_id: 2, creator_id: 1, link_id: 1, link_type: "Following", created_at: "2013-10-31 02:36:41", updated_at: "2013-10-31 02:36:41">, #<Activity id: 3, receiver_id: 2, creator_id: 1, link_id: 1, link_type: "Favorite", created_at: "2013-10-31 02:37:04", updated_at: "2013-10-31 02:37:04">]>

至少,这证明了Rails有能力做到这一点。绕过default_scope似乎是告诉Rails你想做什么而不是技术限制的问题。

<强>更新

事实证明,当您将范围块传递给关联并调用该关联时,该块中的self将引用该关系。所以,你可以反思并采取恰当的行动:

class Activity < ActiveRecord::Base
  belongs_to :creator,  class_name: 'Artist', foreign_key: :creator_id,  inverse_of: :created_activities
  belongs_to :receiver, class_name: 'Artist', foreign_key: :receiver_id, inverse_of: :received_activities
  belongs_to :link, -> { self.reflections[:activity].active_record == Rating ? includes(:rating_explanation) : scoped }, polymorphic: true
end

我更新了我的代码以反映(哈哈)这个。

这可以清理。例如,在访问活动链接时,您可能并不总是希望加载rating_explanations。有很多方法可以解决这个问题。如果你愿意,我可以发一个。

但是,我认为最重要的是,在关联的范围块中,您可以访问正在构建的ActiveRecord::Relation。这将允许您有条件地做事。

答案 1 :(得分:0)

试试这个:

@artist.activities.includes([{:link => :post},:creator,:receiver])...

See the Rails docs for more.

答案 2 :(得分:0)

如果我做对了,你不仅要加载多态关联的关联,还要加载多态关联的多态关联

这基本上意味着通过一堆来自数据库中某些字段的未定义表将一个已定义的表与另一个已定义的表连接起来。

由于活动和帖子都是通过多态对象连接的,因此它们都有something_idsomething_type列。

现在我认为有效记录不会让你开箱即用,但基本上你想要这样的东西:

class Activity
  has_one :post, :primary_key => [:link_id, :link_type], :foreign_key => [:postable_id, :postable_type]
end

(假设Post上的多态关联为belongs_to :postable

然后,这会在PostActivity之间为您提供一种直接关联的查询,这对habtm非常奇怪,具有'多态连接表'(我刚刚制作)那个词)。因为两者共享相同的多态关联对象,所以它们可以连接。有点像我朋友的朋友。

CAVEAT :就像我说的那样,据我所知,AR不会让你开箱即用(虽然它会很棒)但是有一些宝石会给你复合外键和/或主键。也许你可以找到一个帮助你以这种方式解决问题。

答案 3 :(得分:0)

鉴于您对问题的更新(使用AR的include这是不可行的),您可以尝试以下方法通过一个额外的查询来获取相关的帖子:

  • 获取活动,然后构建其链接的post_ids的数组。

    @activities = @artist.activities.includes([:link,:creator,:receiver])
    activity_post_ids = @activities.map{|a| a.link.post_id}.compact
    
  • 然后在一个查询中加载帖子并将其存储在Hash中,并按id

    索引
    activity_posts = Hash[Post.where(id: activity_post_ids).map{|p| [p.id, p]}]
    #=> {1 => #<Post id:1>, 3 => #<Post id:3>, ... }
    
  • 最后循环@activities并设置每个关联链接的post属性

    @activities.each{|a| a.link.post = activity_posts[a.link.post_id]}
    

答案 4 :(得分:0)

这个简单的ActiveRecord应该有效:

@artist.activities.includes([:link => :post,:creator,:receiver]).order("id DESC").limit(30)

如果没有,如果您收到类似"Association named 'post' was not found; perhaps you misspelled it?"的错误,那么您遇到了模型问题,因为至少有一个link关联没有post关联。

多态关联旨在与通用接口一起使用。如果您要求post获取任何多态关联,那么您的所有关联也应该实现该关联(post一个)。

检查多态关联中使用的所有模型是否实现了帖子关联。