我想对包括所有属性的对象进行深层复制。
experiment_old有10个试验。我希望通过10次试验将所有内容复制到experiment_new中。 experiment_old还应该保留10个试用信息。
但是,我在下面尝试过的所有案例,都会很好地复制一切,但是experiment_old没有10个试验信息。它们只是出现在experiment_new上。
为这些案例进行深层复制的最佳方法是什么。
案例1:
@experiment_new = Experiment.create(@experiment_old.attributes.merge(:trials => experiment_old.trails))
案例2:
@experiment_new = Marshal.load(Marshal.dump(@experiment_old.trials))
案例3:
@experiment_new = @experiment_old.clone
以下是模型:
class Experiment < ActiveRecord::Base
belongs_to :experimenter
has_many :trials
has_many :participants
end
class Trial < ActiveRecord::Base
belongs_to :experiment
belongs_to :datum
belongs_to :condition
has_one :result_trial
end
答案 0 :(得分:24)
您可以享受ActiveRecord 3.2的Amoeba gem。
它支持has_one
,has_many
和has_and_belongs_to_many
关联的简单和自动递归复制,字段预处理以及高度灵活且功能强大的配置DSL,可以应用于模型和苍蝇。
请务必查看Amoeba Documentation,但使用非常简单......
刚
gem install amoeba
或添加
gem 'amoeba'
到您的Gemfile
然后将变形虫块添加到模型中并照常运行dup
方法
class Post < ActiveRecord::Base
has_many :comments
has_and_belongs_to_many :tags
amoeba do
enable
end
end
class Comment < ActiveRecord::Base
belongs_to :post
end
class Tag < ActiveRecord::Base
has_and_belongs_to_many :posts
end
class PostsController < ActionController
def some_method
my_post = Post.find(params[:id])
new_post = my_post.dup
new_post.save
end
end
您的新帖子应该包含最初与之关联的所有标记,并且所有评论也应该重复。您可以通过DSL禁用各种记录的复制,您可以在文档中阅读这些记录,但是,例如,如果您想保留标记,而不是注释,则可以执行以下操作:
class Post < ActiveRecord::Base
has_many :comments
has_and_belongs_to_many :tags
amoeba do
include_field :comments
end
end
或使用专有语法
class Post < ActiveRecord::Base
has_many :comments
has_and_belongs_to_many :tags
amoeba do
exclude_field :comments
end
end
或指定要识别(并因此复制)的字段类型
class Post < ActiveRecord::Base
has_many :comments
has_and_belongs_to_many :tags
amoeba do
recognize :has_and_belongs_to_many
end
end
这些不同的选项中的每一个都应该使新帖子与旧帖子重新关联,但不会重复评论。
如果您启用了Amoeba,Amoeba也会自动递归到子记录中
class Post < ActiveRecord::Base
has_many :comments
amoeba do
enable
end
end
class Comment < ActiveRecord::Base
belongs_to :post
has_many :ratings
amoeba do
enable
end
end
class Rating < ActiveRecord::Base
belongs_to :comment
end
您还可以使用一些额外数据为字段添加前缀以指示唯一性
class Post < ActiveRecord::Base
has_many :comments
amoeba do
enable
prepend :title => "Copy of "
end
end
除了前置之外,您还可以在给定字段上附加或运行正则表达式
享受! :)
答案 1 :(得分:22)
您应该克隆每个试验并将它们分配给克隆的实验:
@experiment_new = @experiment_old.clone
@experiment_old.trials.each do |trial|
@experiment_new.trials << trial.clone
end