【Rails学习笔记】用户关注功能(四)
最后一节是用来实现动态列表。
feed稍作改进:
class User < ActiveRecord::Base . . . def feed Micropost.from_users_followed_by(self) end . . . end
from_users_followed_by 方法的初步定义:
class Micropost < ActiveRecord::Base . . . def self.from_users_followed_by(user) followed_user_ids = user.followed_user_ids where("user_id IN (?) OR user_id = ?", followed_user_ids, user) end end
在某些程序中,这样的初步实现已经能满足大部分需求了。不过这不是我们最终要使用的实现方式,当动态列表中的微博数量很多时性能就会下降,这种情况可能会发生在用户关注了 5000 个用户后。本小节,我们会重新实现动态列表,在关注用户数量很多时,性能表现也会很好。
class Micropost < ActiveRecord::Base belongs_to :user default_scope -> { order(created_at DESC) } validates :content, presence: true, length: { maximum: 140 } validates :user_id, presence: true # Returns microposts from the users being followed by the given user. def self.from_users_followed_by(user) followed_user_ids = "SELECT followed_id FROM relationships WHERE follower_id = :user_id" where("user_id IN (#{followed_user_ids}) OR user_id = :user_id", user_id: user.id) end end
这段代码结合了 Rails、Ruby 和 SQL 的优势,达到了目的,而且做的很好。当然,子查询也不是万能的。对于更大型的网站而言,可能要使用“后台作业(background job)”异步生成动态列表。
最后我们要在home中加入对分页功能的支持
class StaticPagesController < ApplicationController def home if signed_in? @micropost = current_user.microposts.build @feed_items = current_user.feed.paginate(page: params[:page]) end end . . . end
下一篇:
DBeaver 超级详细的安装与使用