我遇到了一些我不理解如何使用Rails关联建模的东西,STI和多态似乎都没有解决它。
我希望能够通过has_many:through创建的集合访问连接表中的属性。
在下面的代码中,这意味着我希望能够通过.members集合中的对象访问委员会职位的名称和描述,但据我所知,我不能这样做。我必须通过原始的连接表。
e.g。为俱乐部及其委员会成员建模
class User < ActiveRecord::Base
attr_accessible :full_name,
:email
has_many: committee_positions
has_many: committees, :through => committee_positions
end
class Committee < ActiveRecord::Base
attr_accessible :name
has_many :committee_positions
has_many :members, :through => :committee_positions
end
class CommitteePosition < ActiveRecord::Base
attr_accessible :user_id,
:committee_id,
:member_description,
:role_title
belongs_to :committee
belongs_to :user
end
假设每个委员会职位实例都有唯一的描述 即该描述特别适用于会员和委员会,因此必须存储在联接表中,而不是存储在用户或俱乐部中。
e.g。
Committee member: Matt Wilkins
Role: "Rowing club president"
Description: "Beats the heart of the rowing club to his own particular drum"
有没有办法通过committee.members集合访问连接表中的数据?
虽然活动记录为我们提供了直接转发给成员的优秀别名,但似乎没有办法访问创建该集合的连接表上的数据:
我无法做到以下几点:
rowing_committee.members.find_by_role_title('president').name
.members集合中的每个项目都是一个用户对象,似乎无法访问存储在CommissionPositions连接表中的角色或描述。
这样做的唯一方法是:
rowing_committee.committee_positions.find_by_role_title('president').user.name
这是完全可行的但是笨重而且无益。我觉得用例足够通用,我可能会遗漏一些东西。
我想通过委员会收集的内容访问<。strong>
member
- full_name
- email
- role_title (referenced from join table attributes)
- member_description (referenced from join table attributes)
这只是一件小事,但感觉很难看。是否有一种干净的方法来指示“成员”对象继承连接表中包含的信息?
在完成这项工作后,我意识到我可以通过简单地为委员会成员定义一个新类并在has_many:through关系中引用而不是用户来解决问题。它工作得更好但仍然很笨重
class Committee < ActiveRecord::Base
...
has_many :committee_positions
has_many :members,
:through => :committee_positions,
:class_name => 'CommitteeMember'
...
end
class CommitteeMember < User
def description( committee )
self.committees.find_by_committee_id( committee.id ).description
end
def role( committee )
self.committees.find_by_committee_id( committee.id ).description
end
end
现在这种情况越来越近了但是使用它的代码仍然是笨重的:
committee = Committee.first
president_description = committee.members.find_by_title('president').description( committee )
有没有办法用他们引用的委员会初始化这些对象?
答案 0 :(得分:1)
我想你可以在这里使用一些代表团。在您的Committee_Position类中:
class Committee_Position < ActiveRecord::Base
attr_accessible :user_id,
:committee_id,
:member_description,
:role_title
belongs_to :committee
belongs_to :user
delegate :name, :email, :to => :user
end
所以你可以做你想说的话:
rowing_club.committee_members.find_by_role_title('president').name