4

我有一个带有以下(简化)模型的 Rails 应用程序:

# member.rb
class Member < ActiveRecord::Base
  has_many :member_roles, :dependent => :destroy, 
           :autosave => true, :inverse_of => :member
end

...

# member_role.rb
class MemberRole < ActiveRecord::Base
  belongs_to :member, :inverse_of => :member_roles
  validates_presence_of :member_id
end

如果我尝试.build在关联上使用该方法,则创建的对象没有设置外键。这会导致它验证失败,或者没有验证就无法与Member.

# Rails console
> m = Member.find(280)
> mr = m.member_roles.build(:role_id => Role.find_by_name("Crew Chief").id)
=> #<MemberRole id: nil, member_id: nil, role_id: 6697350, start_date: nil, \
   end_date: nil, memo: nil, created_at: nil, updated_at: nil>
> mr.save!
ActiveRecord::RecordInvalid: Validation failed: Member can't be blank
> mr.save(:validate => false)
> mr
=> #<MemberRole id: 1834, member_id: nil, role_id: 6697350, start_date: nil, \
   end_date: nil, memo: nil, created_at: "2012-04-11 06:37:00", \
   updated_at: "2012-04-11 06:37:00">

这与Rails 指南冲突:

collection.build 方法返回关联类型的一个或多个新对象。这些对象将从传递的属性中实例化,并通过它们的外键创建链接,但关联的对象尚未保存。

显然,手动设置 member_id 很容易解决。但我想避免这种情况。我相信这段代码在以前版本的 rails 中可以正常工作;以上是 Rails 3.2.3 的行为。

4

1 回答 1

0

我将粗略回答中的代码作为初始值设定项:可以将嵌套属性与继承结合使用吗?

将其更改为:

class ActiveRecord::Reflection::AssociationReflection
  def build_association(*options, &block)
    if options.first.is_a?(Hash) and options.first[:type].presence
      options.first[:type].to_s.constantize.new(*options, &block)
    else
      klass.new(*options, &block)
    end
  end
end

解决了这个问题。

于 2012-04-11T21:12:27.780 回答