我想做一个ActiveRecord对象的副本,改变进程中的一个字段(除了id)。要做到这一点,最简单的方法是什么?

我意识到我可以创建一个新记录,然后遍历每个字段,逐字段复制数据—但我认为一定有更简单的方法来做到这一点。

也许是这样的:

 new_record = Record.copy(:id)

当前回答

要获得副本,使用dup(或clone for < rails 3.1+)方法:

#rails >= 3.1
new_record = old_record.dup

# rails < 3.1
new_record = old_record.clone

然后您可以更改任何您想要的字段。

ActiveRecord覆盖了内置的object#克隆,为您提供了一个新的(未保存到DB)记录,该记录具有未分配的ID。 请注意,它不会复制关联,因此如果需要,您必须手动执行此操作。

Rails 3.1克隆是一个浅拷贝,使用dup代替…

其他回答

简单的方法是:

#your rails >= 3.1 (i was done it with Rails 5.0.0.1)
  o = Model.find(id)
 # (Range).each do |item|
 (1..109).each do |item|
   new_record = o.dup
   new_record.save
 end

Or

# if your rails < 3.1
 o = Model.find(id)
 (1..109).each do |item|
   new_record = o.clone
   new_record.save
 end     

我通常只是复制属性,改变任何我需要改变的:

new_user = User.new(old_user.attributes.merge(:login => "newlogin"))

因为可能会有更多的逻辑,当复制一个模型时,我建议创建一个新的类,在那里你可以处理所有需要的逻辑。 为了缓解这种情况,有一种宝石可以帮助你:小丑

根据他们的文档示例,对于User模型:

class User < ActiveRecord::Base
  # create_table :users do |t|
  #  t.string :login
  #  t.string :email
  #  t.timestamps null: false
  # end

  has_one :profile
  has_many :posts
end

你创建你的克隆类:

class UserCloner < Clowne::Cloner
  adapter :active_record

  include_association :profile, clone_with: SpecialProfileCloner
  include_association :posts

  nullify :login

  # params here is an arbitrary Hash passed into cloner
  finalize do |_source, record, params|
    record.email = params[:email]
  end
end

class SpecialProfileCloner < Clowne::Cloner
  adapter :active_record

  nullify :name
end

然后使用它:

user = User.last
#=> <#User(login: 'clown', email: 'clown@circus.example.com')>

cloned = UserCloner.call(user, email: 'fake@example.com')
cloned.persisted?
# => false

cloned.save!
cloned.login
# => nil
cloned.email
# => "fake@example.com"

# associations:
cloned.posts.count == user.posts.count
# => true
cloned.profile.name
# => nil

从项目中复制的例子,但它会给你一个清晰的愿景,你可以实现什么。

简单来说,我想说的是:

Model.new (Model.last.attributes。拒绝{|k,_v| k.to_s == 'id'}

根据您的需要和编程风格,还可以结合使用类的新方法和merge。由于缺乏更好的简单示例,假设您有一个计划在某个日期执行的任务,并且希望将其复制到另一个日期。任务的实际属性并不重要,所以:

old_task = Task.find(task_id)
new_task = Task.new(old_task.attributes.merge({:scheduled_on => some_new_date}))

将创建一个新任务:id => nil,:scheduled_on => some_new_date,所有其他属性与原始任务相同。使用任务。new,你必须显式调用save,所以如果你想自动保存它,改变任务。new to Task.create。

和平。

如果你不想复制id,使用ActiveRecord::Base#dup