意外的切片追加行为
问题介绍
将元素追加到切片内的切片时循环,随后尝试从循环结果创建新切片会出现意外行为。具体来说,最后一个追加元素会覆盖先前追加的结果。当基于循环迭代获得的切片创建新切片时,会出现此问题。
分析
观察到的行为源于 Go 切片和数组都使用的事实相同的底层数据结构。当附加到切片时,Go 创建一个引用相同底层数组的新切片对象。这意味着对原始切片所做的任何更改也会影响新切片。
解决方案
为了避免这种行为,必须在制作之前复制切片任何修改。这将创建一个新的切片对象,该对象具有对底层数组的独立引用。实现此目的的惯用方法是使用复制函数:
<code class="go">func makeFromSlice(sl []int) []int { result := make([]int, len(sl)) copy(result, sl) return result }</code>
通过将 makeFromSlice(i) 的结果分配给主函数中的 j,我们创建了一个独立于原始 i 的新切片片。这个新切片可以安全地修改,而不影响原始切片。
与切片文字对比
切片文字,例如 {0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10},当追加的数组超出其容量时,总是分配一个新的底层数组。这解释了为什么 sliceFromLiteral 的行为符合预期。
结论
使用切片时,请始终记住在修改它们之前先进行复制。这确保了对新切片所做的更改不会影响原始切片。复制功能提供了一种惯用且有效的方法来执行此操作。
以上是为什么从循环创建新切片时切片追加行为会发生变化?的详细内容。更多信息请关注PHP中文网其他相关文章!