最后一节是用来实现动态列表。
在上一章我们已经初步实现了动态列表,但那时候我们还没有实现用户之间的关注功能,显示的只是自己的微博,这里我们要改进一下,使其实现自己和所关注用户的微博。
举个例子:id 为 1 的用户关注了 id 为 2,7,8,10 的用户后得到的动态列表
feed稍作改进:
class User < ActiveRecord::Base . . . def feed Micropost.from_users_followed_by(self) end . . . end
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
上面所用代码的问题在于followed_user_ids = user.followed_user_ids 这行代码,它会把所有被关注用户的 id 取出存入内存,然后再创建一个元素数量和被关注用户数量相同的数组。既然它的目的只是为了检查集合是否包含了指定的元素,那么就一定存在一种更高效的方法,其实 SQL 真的提供了针对这种问题的优化措施:使用子查询把查询被关注用户 id 的操作放入数据库层进行。
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
最后我们要在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