切片追加中的意外行为:如何在不考虑修改的情况下创建多个切片
在 Go 代码中操作切片时,你可能会遇到一个令人困惑的问题问题:在循环中将元素追加到切片,然后使用循环结果创建新切片可能会导致最后追加覆盖之前附加的切片。出现此行为的原因是切片引用相同的基础数组值。
示例:
<code class="go">func create(iterations int) []int { a := make([]int, 0) for i := 0; i < iterations; i++ { a = append(a, i) } return a } func sliceFromLoop() { i := create(11) j := append(i, 100) g := append(i, 101) h := append(i, 102) fmt.Printf("i: %v\nj: %v\ng: %v\nh:%v\n", i, j, g, h) }</code>
在此示例中,sliceFromLoop 函数创建一个切片 i 并附加不同的切片值,产生切片 j、g 和 h。然而,所有三个切片都指向同一个底层数组,因此当最后一个追加修改数组时,它会影响所有切片。
解决方案:复制切片进行独立修改
基于现有切片创建多个切片并避免修改问题的惯用方法是在附加任何内容之前复制切片。这确保每个新切片都有自己的底层数组。
<code class="go">func makeFromSlice(sl []int) []int { result := make([]int, len(sl)) copy(result, sl) return result }</code>
用法示例:
<code class="go">func main() { i := make([]int, 0) for ii := 0; ii < 11; ii++ { i = append(i, ii) } j := append(makeFromSlice(i), 100) // works fine }</code>
在这个修改后的示例中,我们创建了我在附加 100 之前进行切片。这可以确保 j 引用单独的底层数组,并且不会受到将来对 i 的修改的影响。
切片文字行为的说明
此问题发生的原因不会出现在切片文字中(例如, i := []int{0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10})是,如果追加操作超出后备数组的容量,则分配一个新数组。此行为与切片文字无关,是附加到切片的基本属性。
以上是为什么附加到循环中的切片会影响 Go 中的其他切片?的详细内容。更多信息请关注PHP中文网其他相关文章!