为什么我的副本也会影响原件?

How come my duplicate also affects the original?

我正在尝试将副本添加到经过一些修改的数组中:

  site.pages.dup.each do | page |
    new_page = page.dup
    data = new_page.data
    data['permalink'] = File.join('/app', page.url)
    data['layout'] = 'app'
    site.pages << new_page
  end

  site.pages.each do | page |
    puts page.data
  end 

当我输出 page.data 时,永久链接不同,但 data['layout'] 对数组中的所有项目都是相同的。我还尝试了 data = new_page.data.dup 并在每个实例中写出 new_page.data

这里的问题是 dup 只是一个浅拷贝。这意味着它创建了第一层的副本,即数组本身,而不是任何更深的层。

您可能需要考虑像 full_dup 这样的 gem,它添加了 full_dup 方法,可以根据需要深入挖掘以创建数据的完整副本。

完全披露:我写了 full_dup gem。可能还有其他 gem 也同样有效。

(移动评论以回答格式问题。)

请提供数据示例;我在使用类似结构时遇到的问题为零,例如

pages = [
  {
    'name' => 'page 1',
    'data' => {
      'permalink' => 'p1 perma',
      'layout'    => 'p1 layout'
    }
  },

  {
    'name' => 'page 2',
    'data' => {
      'permalink' => 'p2 perma',
      'layout'    => 'p2 layout'
    }
  },
]

如果我(基本上)使用你的代码,加上我的评论:

pages.dup.each do |p|
  new_page = p.dup

  new_data = new_page['data'].dup

  new_data['layout']    = 'app'
  new_data['permalink'] = "#{new_data['permalink']} whatever"

  new_page['data'] = new_data

  pages << new_page
end

我得到以下输出:

[{"name"=>"page 1", "data"=>{"permalink"=>"p1 perma", "layout"=>"p1 layout"}},
 {"name"=>"page 2", "data"=>{"permalink"=>"p2 perma", "layout"=>"p2 layout"}},
 {"name"=>"page 1", "data"=>{"permalink"=>"p1 perma whatever", "layout"=>"app"}},
 {"name"=>"page 2", "data"=>{"permalink"=>"p2 perma whatever", "layout"=>"app"}}]

无关,但我会分开数组连接;相反,请考虑 mapping 页面并使用原始 pages 和修改后的数组创建一个新数组。现在你正在做一个不必要的额外 dup,它增加了一些混乱。

我还会将 data dup/modification 分解成一个单独的方法(可能还有整个重复过程)以真正保持紧凑,粗略地说:

def modify_page_data(data)
  new_data = data.dup

  new_data['permalink'] = "#{data['permalink']} whatever"
  new_data['layout'] = 'app'

  new_data
end

def dup_page(page)
  new_page = page.dup
  new_page['data'] = modify_page_data(page['data'])
  new_page
end

new_pages = pages.map(&method(:dup_page))

然后您可以 concat 或任何您想要的原始 pages 页面和修改后的新页面。

https://github.com/davelnewton/Whosebug/tree/master/ruby/51289405