Rails ActiveRecord Scope是另一个范围的“对立面”,或者是“缺少”属性的用户

时间:2013-01-18 19:56:50

标签: ruby-on-rails-3 activerecord scope named-scope

我有一个user.rb的模型,我在其中为admins定义了一个范围,这是通过权限表扮演管理员角色的用户。

has_many :permissions
has_many :roles, :through => :permissions

范围的工作原理如下:

scope :admins, joins(:permissions).merge(Permission.admin_permissions)

我还想制作一个名为non-admins的范围或类似的范围,这是所有没有管理员角色的用户。

最简单的方法是什么?

2 个答案:

答案 0 :(得分:6)

如果您想要反向SQL查询,则必须手动完成。没有内置的ActiveRecord解决方案。

scope :admins, joins(:permissions).merge(Permission.where("permissions.admin = true"))
scope :non_admins, joins(:permissions).merge(Permission.where("permissions.admin = false"))

如果有很多范围或它们很复杂,请考虑用id排除它们:

User.where("id not in (?)", User.admins.pluck(:id))

# or if you are already using admin records
admins = User.admins
User.where("id not in (?)", admins.map(&:id))

根据行数和原始查询的复杂程度,这可能比以前的方式更慢或更快。

答案 1 :(得分:0)

对于很多用户来说,不具备高效率的简单方法:

admin_user_ids  = User.admins.map(&:id)
non_admin_users = User.all.reject { |u| admin_user_ids.include?(u.id) }