如何在Rails 3中将多个属性传递给find_or_create_by?

时间:2010-10-22 13:03:27

标签: ruby-on-rails ruby-on-rails-3

我想使用find_or_create_by,但此语句不起作用。它不会与其他属性“找到”或“创建”。

productproperty = ProductProperty.find_or_create_by_product_id(:product_id => product.id, :property_id => property.id, :value => d[descname])

关于在Rails 3中使用动态查找器的信息似乎很少或没有。“和” - 这些一起给我一个未知的方法错误。

更新:

最初我无法让以下工作。请假设我不是白痴,“产品”是产品AR模型的一个实例。

product.product_properties.find_or_create_by_property_id_and_value(:property_id => 1, :value => "X")

错误方法是:

no such keys: property_id, value

我无法理解这一点。只有今天早上我找到了传递这样的值的参考:

product.product_properties.find_or_create_by_property_id_and_value(1, "X")

瞧,它运作正常。我原本期望哈希在相同的情况下工作,但我猜不是。

所以如果你错过互联网上的某些内容,我猜你会得到一次投票?

5 个答案:

答案 0 :(得分:52)

如果要按多个属性进行搜索,可以使用“和”来追加它们。例如:

productproperty = ProductProperty.find_or_create_by_product_id_and_property_id_and_value(:product_id => product.id, :property_id => property.id, :value => d[descname])

有一个小问题需要注意。它将始终返回您指定的对象,即使由于验证错误而无法保存该对象。因此,请确保检查返回的对象是否具有id(或is_valid?)。不要在数据库中假设它。

或者,如果无法保存对象,您可以使用该方法的“bang”版本引发错误:

http://guides.rubyonrails.org/active_record_querying.html#find-or-create-by-bang

这适用于Rails 3。

答案 1 :(得分:23)

请参阅http://api.rubyonrails.org/classes/ActiveRecord/Base.html

使用单个查询参数:

productproperty = ProductProperty.find_or_create_by_product_id(product.id) { |u| u.property_id => property_id, u.value => d[descname] } )

或使用多个参数进行扩展:

productproperty = ProductProperty.find_or_create_by_product_id(:product_id => product.id, :property_id => property_id, :value => d[descname]) { |u| u.property_id => property_id, u.value => d[descname] } )

<击>

可以使用:

conditions = { :product_id => product.id, 
               :property_id => property.id,
               :value => d[descname] }

pp = ProductProperty.find(:first, :conditions => conditions) || ProductProperty.create(conditions) 

答案 2 :(得分:10)

您也可以使用where(...).first_or_create - ActiveRecord::Relation#first_or_create

product_property_attrs  = { product_id: product.id, 
                            property_id: property.id, 
                            value: d[descname] }

product_property = ProductProperty.where(product_property_attrs).first_or_create

答案 3 :(得分:9)

我在Rails 3.1中发现,您不需要将属性作为哈希传递。您只需传递值。

ProductProperty.find_or_create_by_product_id_and_property_id_and_value(
  product.id, property.id, d[descname])

答案 4 :(得分:7)

在Rails 4中,您可以使用find_or_create_by(attr1: 1, attr2: 2)来查找或创建多个属性。

您还可以执行以下操作:

User.create_with(
  password: 'secret', 
  password_confirmation: 'secret',
  confirmation_date: DateTime.now
).find_or_create_by(
  email: 'admin@domain.com',
  admin: true
)

如果您需要使用某些属性创建用户,但无法按这些属性进行搜索。