Golang中range循环的行为解析与使用帮助
Golang中range循环的行为解析与使用帮助
以下代码的行为会根据我使用 for i := range list 后接 x := list[i] 还是使用 for _, x := range list 而有所不同。我猜测这是循环变量在闭包中被意外捕获的一种略显奇怪的表现形式,不过我想看看是否有其他人能更好地解释这一点。
package main
import (
"fmt"
)
type Log struct {
Log *string
}
func main() {
resp := []string{"string1", "string2"}
var logs []*string
// 取消注释以下行以获得预期行为
//for i := range resp {
// h := resp[i]
for _, h := range resp {
logs = append(logs, &h)
}
for _, l := range logs {
fmt.Println(*l)
}
}
更多关于Golang中range循环的行为解析与使用帮助的实战教程也可以访问 https://www.itying.com/category-94-b0.html
2 回复
package main
import (
"fmt"
)
type Log struct {
Log *string
}
func main() {
resp := []string{"string1", "string2"}
var logs []*string
// 循环的每次迭代都使用同一个范围变量 h 的实例,
// 因此每个闭包都共享这个单一变量。
for _, h := range resp {
h := h // 新的局部变量
logs = append(logs, &h)
}
for _, l := range logs {
fmt.Println(*l)
}
}
https://play.golang.org/p/SnETZW17YOh
string1
string2
更多关于Golang中range循环的行为解析与使用帮助的实战系列教程也可以访问 https://www.itying.com/category-94-b0.html
这是一个典型的循环变量捕获问题,在Go的range循环中确实需要注意。两种写法的区别在于变量作用域和内存地址的分配。
问题分析
在 for _, h := range resp 中,h 是单个循环变量,在每次迭代中会被重新赋值,但其内存地址保持不变:
// 实际执行类似这样:
var h string // 只在循环开始时声明一次
for i := 0; i < len(resp); i++ {
h = resp[i] // 每次迭代重新赋值
logs = append(logs, &h) // 每次都添加同一个地址
}
所有 logs 中的指针都指向同一个 h 变量,最终 h 的值是最后一次迭代的值(“string2”)。
解决方案
方案1:使用索引创建局部变量
for i := range resp {
h := resp[i] // 每次迭代创建新的局部变量
logs = append(logs, &h)
}
方案2:直接获取元素地址
for i := range resp {
logs = append(logs, &resp[i]) // 直接使用原切片的元素地址
}
方案3:使用匿名函数创建新作用域
for _, h := range resp {
h := h // 创建局部副本
logs = append(logs, &h)
}
完整示例
package main
import "fmt"
func main() {
resp := []string{"string1", "string2"}
// 正确的方式1:使用索引
var logs1 []*string
for i := range resp {
h := resp[i]
logs1 = append(logs1, &h)
}
// 正确的方式2:直接取地址
var logs2 []*string
for i := range resp {
logs2 = append(logs2, &resp[i])
}
// 正确的方式3:创建局部副本
var logs3 []*string
for _, h := range resp {
h := h
logs3 = append(logs3, &h)
}
fmt.Println("方式1输出:")
for _, l := range logs1 {
fmt.Println(*l) // 输出: string1, string2
}
fmt.Println("方式2输出:")
for _, l := range logs2 {
fmt.Println(*l) // 输出: string1, string2
}
fmt.Println("方式3输出:")
for _, l := range logs3 {
fmt.Println(*l) // 输出: string1, string2
}
}
关键点
for _, x := range中的x是复用的变量,不是每次迭代新建- 当需要保留值的引用时,必须确保每个引用指向不同的内存位置
- 这个问题在goroutine闭包中更为常见,原理相同
这种设计是为了性能优化,避免每次迭代都分配新变量,但在需要捕获变量时需要注意作用域问题。

