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"

ここで何が欠けているのですか?!?! 2 番目の #T を追加できるのは一体何でしょう?

4

1 に答える 1

1

紛らわしいことに (少なくとも私には) after_initialize、新しいインスタンスを作成した後だけでなく、データベースから既存のインスタンスをロードした後も、アクティブなレコード オブジェクトがインスタンス化されるたびに呼び出されます。したがって、実行時に 2 番目の 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 に答える