有没有办法来加载多态关联的关联?
艺术家有很多活动(基本上是用户之间的交互缓存):
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_ids中的所有帖子选中。 然后以某种方式将'post'结果注入到@activities中,以便在遍历集合时作为activity.link.post提供。 可能?
TL; DR我的解决方案使artist.created_activities.includes(:link)
急切地加载你想要的一切
这是我的第一次尝试:https://github.com/JohnAmican/music
一些注意事项:
default_scope
,所以这不是最优的。 activities
的艺术家; 你必须引用created_activities
或received_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
。 这将允许你有条件地做事情。
尝试这个:
@artist.activities.includes([{:link => :post},:creator,:receiver])...
有关更多信息,请参阅Rails文档。
如果我明白了这一点,你不仅需要加载一个多态关联的关联 ,而且还要加载一个多 态关联的多态关联 。
这基本上意味着通过来自数据库中某些字段的一堆未定义的表,将一个定义的表与另一个定义的表连接起来。
由于活动和帖子都通过多态对象连接,因此它们都有一个something_id
和something_type
列。
现在我认为积极的记录不会让你这样做,但基本上你想要的东西是这样的:
class Activity
has_one :post, :primary_key => [:link_id, :link_type], :foreign_key => [:postable_id, :postable_type]
end
(假设Post
上的多态关联belongs_to :postable
)
然后这会给你一个查询排序 - Post
和Activity
之间的直接关联,这是一个非常奇怪的带有'多态连接表'(我刚刚提出这个词)的habtm
。 因为它们共享相同的多态关联对象,所以它们可以连接。 有点像朋友的朋友的东西。
CAVEAT :就像我所说的,据我所知,AR不会让你这样做(虽然它会很棒),但是有一些宝石会给你复合的外键和/或主键。 也许你可以找到一个帮助你以这种方式解决你的问题。
链接地址: http://www.djcxy.com/p/56933.html上一篇: is there a way to eager load polymorphic association's associations?