共同的朋友作为活动记录阿雷尔关系



我为我的用户提供了以下模型:

class User < ActiveRecord::Base
  has_many :facebook_friendships
  has_many :facebook_friends, :through => :facebook_friendships, :source => :friend
  def mutual_facebook_friends_with(user)
    User.find_by_sql ["SELECT users.* FROM facebook_friendships AS a
                        INNER JOIN facebook_friendships AS b
                          ON a.user_id = ? AND b.user_id = ? AND a.friend_id = b.friend_id
                        INNER JOIN users ON users.id = a.friend_id", self.id, user.id]
  end
end
class FacebookFriendship < ActiveRecord::Base
  belongs_to :user
  belongs_to :friend, :class_name => 'User'
end
如果 ID 为 53 的用户和 ID 为 97 的用户

彼此是好友,则数据库中的 facebook_friendships 表中将有行 [53, 97] 和 [97, 53]。这是我想出的用于计算共同朋友的原始sql查询:

SELECT users.* FROM facebook_friendships AS a
  INNER JOIN facebook_friendships AS b
    ON a.user_id = :user_a AND b.user_id = :user_b AND a.friend_id = b.friend_id
  INNER JOIN users ON users.id = a.friend_id

我mutual_friends_with返回关系而不是数组。这样,我可以将结果与其他条件链接起来,例如 where(college:"NYU"),并获得所有 ActiveRecord 的优点。有没有好方法可以做到这一点?

你试过#find_by_sql吗?

http://guides.rubyonrails.org/active_record_querying.html#finding-by-sql

如果您想使用自己的 SQL 在表中查找记录,您可以 使用find_by_sql。find_by_sql方法将返回一个数组 对象,即使基础查询仅返回单个记录。为 示例,您可以运行此查询:

 Client.find_by_sql("SELECT * FROM clients 
                     INNER JOIN orders ON clients.id = orders.client_id   
                     ORDER clients.created_at desc")

find_by_sql为您提供了一种进行自定义调用的简单方法 数据库和检索实例化的对象。

这是我用来结交共同朋友的方法。

has_many :company_friendships, autosave: true
has_many :company_friends, through: :company_friendships, autosave: true
has_many :inverse_company_friendships, class_name: "CompanyFriendship", foreign_key: "company_friend_id", autosave: true
has_many :inverse_company_friends, through: :inverse_company_friendships, source: :company, autosave: true  
def mutual_company_friends
  Company.where(id: (company_friends | inverse_company_friends).map(&:id))
end

最新更新