为什么我的副本也会影响原件?
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"}}]
无关,但我会分开数组连接;相反,请考虑 map
ping 页面并使用原始 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
我正在尝试将副本添加到经过一些修改的数组中:
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"}}]
无关,但我会分开数组连接;相反,请考虑 map
ping 页面并使用原始 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