1

这简直是​​疯了。我有一个简单的回调函数来初始化和验证一个类子类:

class A < ActiveRecord::Base
   has_many :bs
   after_initialize :add_t_instance
   validate :has_only_one_t

  protected

  def add_t_instance
      bs << B.new(:a => self, :type => "T") unless bs.map(&:type).count("T") > 0
  end

  def has_only_one_t
     unless bs.map(&:type).count("T") < 2
       errors.add(:bs, 'has too many Ts")
     end
  end

end

现在,运行时的魔法来了:

a = A.new
>>[#<A>]
a.bs
>> [#<T>]
a.save
>> true
a.id
>> 15

到目前为止一切都很好,但是:

s = A.find(15)
s.bs
>>[#<T>,#<T>]
s.bs.count
>> 2
s.valid?
>> false
s.errors.full_messages
>> "Too many Ts"

我到底在这里想念什么?!?!世界上有什么可以添加第二个#T?

4

1 回答 1

1

令人困惑的是(至少对我而言)after_initialize每当实例化活动记录对象时都会调用,不仅在创建新实例之后,而且在从数据库加载现有实例之后。因此,您在运行时创建了第二个 B A.find(15)

您可以通过检查您是否正在处理回调中的新记录来解决问题,例如

def add_t_instance
  if new_record?
    bs << B.new(:a => self, :type => "T") unless bs.map(&:type).count("T") > 0
  end
end

或者您可以在before_initialize声明本身上设置一个条件,或者尝试使用 before_create 回调。

于 2013-03-11T18:59:33.000 回答