获取具有书签状态的记录列表



我有一个应用程序,用户可以在其中为帖子添加书签。为了处理书签,我创建了一个名为Bookmark的多态模型。

现在,我需要显示所有帖子的列表(假设每页 50 个,分页(,以及当前登录用户的标题和书签状态。

我脑子里有两个场景,我该如何实现这一目标。而且我不喜欢他们中的任何一个。

  1. 在呈现列表时检查每个帖子的书签状态 - 这会产生 N+1 查询,并且基本上是一个可怕的想法
  2. 获取当前
  3. 页面的帖子ID列表,获取当前用户的所有书签,bookmarkable_id IN post_ids_of_current_page,检查可书签列表中存在哪个帖子,并基于此显示书签状态。

第二个非常合理,但我真的很想将其全部合并到一个SQL查询中(如果可能的话,我不确定是否可以使用joins来完成(。这是因为,事实上,我不是在常规的轨道视图中使用它,而是在 graphql API 端点 + SPA 应用程序中使用它,获取集合 ID 并进行比较并不是那么简单。它只会导致对 API 的大量请求,我希望避免这种情况。

事实证明,我可以为此使用includes方法(这将产生两个查询,但它仍然比 N+1 更好(。如果有人可以与joins共享一个查询版本,我将非常乐意接受它的答案,但到目前为止,我正在发布我的解决方案,这足以满足我的需求。

class User
has_many :bookmarks, as: :bookmarker
end
class Post
has_many :bookmarks_as_bookmarkable, as: :bookmarkable, class_name: :Bookmark
def bookmarked_by?(bookmarker)
self.bookmarks_as_bookmarkable.to_a.select do |b|
b.bookmarker_type == bookmarker.class.name &&
b.bookmarker_id   == bookmarker.id
end.any?
end
end

结果是:

Post.includes(:bookmarks_as_bookmarkable).map { |p| u.bookmarked_by?(current_user) }`  
# SELECT "posts".* FROM "posts"
# SELECT "bookmarks".* FROM "bookmarks" WHERE "bookmarks"."bookmarkable_type" = ? AND "bookmarks"."bookmarkable_id" IN (1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23, 24, 25, 26, 27, 28, 29, 30)
#
# => [false, false, false, false, false, false, false, false, false, false, false, false, false, false, false, false, false, false, false, false, false, false, false, false, false, false, false, false, false, true]

最新更新