在 Rails 3 中过滤 has_many :through 关系中的子对象
您好,
我有一个应用程序,其中 Companies
和 User
需要通过 CompanyMembership
模型相互归属,该模型包含有关成员资格的额外信息(具体来说,用户是否是公司的管理员,通过布尔值admin
)。代码的简单版本:
class CompanyMembership < ActiveRecord::Base
belongs_to :company
belongs_to :user
end
class Company < ActiveRecord::Base
has_many :company_memberships
has_many :users, :through => :company_memberships
end
class User < ActiveRecord::Base
has_many :company_memberships
has_many :companies, :through => :company_memberships
end
当然,这使得通过company.users.all等获取公司的所有成员变得简单。但是,我试图获取公司中作为该公司管理员的所有用户的列表(并测试用户是否是给定公司的管理员)。我的第一个解决方案在 company.rb
中如下所示:
def admins
company_memberships.where(:admin => true).collect do |membership|
membership.user
end
end
def is_admin?(user)
admins.include? user
end
虽然这可行,但感觉有些效率低下(它迭代每个成员资格,每次都执行 SQL,对吗?或者 Relation 比这更聪明吗?),我不确定是否有更好的方法来解决这个问题(也许使用范围或 Rails 3 使用的新奇的 Relation
对象?)。
任何关于最佳进行方式的建议(最好使用 Rails 3 最佳实践)将不胜感激!
Greetings,
I have an application where Companies
and Users
need to belong to each other through a CompanyMembership
model, which contains extra information about the membership (specifically, whether or not the User is an admin of the company, via a boolean value admin
). A simple version of the code:
class CompanyMembership < ActiveRecord::Base
belongs_to :company
belongs_to :user
end
class Company < ActiveRecord::Base
has_many :company_memberships
has_many :users, :through => :company_memberships
end
class User < ActiveRecord::Base
has_many :company_memberships
has_many :companies, :through => :company_memberships
end
Of course, this makes it simple to get all the members of a company via company.users.all
, et al. However, I am trying to get a list of all Users in a Company who are admins of that Company (and also to test whether a user is an admin of a given company). My first solution was the following in company.rb
:
def admins
company_memberships.where(:admin => true).collect do |membership|
membership.user
end
end
def is_admin?(user)
admins.include? user
end
While this works, something feels inefficient about it (it's iterating over each membership, executing SQL each time, right? Or is Relation smarter than that?), and I'm not sure if there's a better way to go about this (perhaps using scopes or the fancy new Relation
objects that Rails 3 uses?).
Any advice on the best way to procede (preferably using Rails 3 best practices) would be greatly appreciated!
如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。
绑定邮箱获取回复消息
由于您还没有绑定你的真实邮箱,如果其他用户或者作者回复了您的评论,将不能在第一时间通知您!
发布评论
评论(5)
我相信我以错误的方式处理这个问题,在
company_memberships
而不是users
上指定条件,这才是我真正想要的(Users
列表) >,而不是CompanyMemberships
列表)。我认为我正在寻找的解决方案是:它生成以下 SQL(对于 ID 为 1 的公司):
我还不确定是否需要它,但是
includes()
方法如有必要,将执行预先加载以减少 SQL 查询的数量:(我'我仍然愿意接受任何认为这不是最好/最有效/正确的方法的人的任何建议。)
I believe I was going about this the wrong way, specifying conditions on
company_memberships
instead ofusers
, which was what I actually wanted (a list ofUsers
, not a list ofCompanyMemberships
). The solution I think I was looking for is:which generates the following SQL (for company with ID of 1):
I'm not sure yet if I'll need it, but the
includes()
method will perform eager loading to keep down the number of SQL queries if necessary:(I'm still open to any suggestions from anyone who thinks this isn't the best/most effective/right way to go about this.)
一种更简洁的方法是将关联添加到您的公司模型中,如下所示:
您可能必须深入研究 rails doc 以获得准确的语法。
您不需要 :include,除非您有与 :user 关联的其他类,您可能会在视图中引用它们。
An even cleaner way would be to add an association to your Company model, something like this:
You might have to dig into the rails doc to get the exact syntax right.
You shouldn't need :include, unless you have other classes associated with :user that you might reference in your view.
像这样的事情怎么样:
How about something like this:
我偶然发现了这个答案,并相信现在有一种更好的方法使用
has_many 关联范围
(has_many 文档):has_many 关联的第二个参数可以是包含过滤器的 proc 或 lambda。
I stumbled across this answer and believe that nowadays there is a nicer way using
has_many association scopes
(has_many documentation):The second parameter of a has_many association can be a proc or lambda that contains your filter.
我制作了 activerecord_where_assoc gem 来执行此操作。
有了它,无需连接或包含。进行急切加载仍然是您的选择,而不是义务。
问题不清楚如何过滤公司,所以我自己添加了。如果没有这个,如果一个用户可以在多个公司工作,那么在一个公司担任管理员可能意味着被视为另一家公司的管理员。
该 gem 在 Rails 3 中不起作用,但 9 年后我们支持 Rails 4.1 及更多版本。
以下是简介和示例。请参阅文档了解更多详细信息。
I made the activerecord_where_assoc gem to do this.
With it, no need for joins or includes. Doing eager loading remains your choice, not an obligation.
The question is unclear on how to filter the company, so I added it myself. Without this, if a user can be in multiple company, it being admin in one company could mean being treated as an admin in another company.
The gem doesn't work in Rails 3, but we are 9 years later and Rails 4.1 and more are supported.
Here are the introduction and examples. Read more details in the documentation.