loopvar 改动不同版本的影响-defer,closures

defer示例代码

go 复制代码
package main

import (
	"fmt"
	"runtime"
)

func main() {
	fmt.Println("golang version:", runtime.Version())
	defer_demo()
}

func defer_demo() {
	defer println()
	for counter, n := 0, 0; n < 3; n++ {
		defer func(v int) {
			fmt.Print(counter)
			counter++
		}(n)
	}
}

代码说明

这个代码先打印了使用的golang的版本,然后进行一个for循环,for循环中定义了一个defer func,打印出循环时定义的变量counter的值,在counter自增 go run xxx.go看下结果

golang 1.21的运行结果

复制代码
go run demo/defer.go
golang version: go1.21.5
012

可以看到在1.21.5这个版本中,打印出来的counter值自增成功了,在1.22中又会如何呢

golang 1.22的运行结果

复制代码
go run demo/defer.go     
golang version: go1.22.1
000

可以看到在1.22.1的这个版本中,打印出来的值都是一样的。

再举个例子

go 复制代码
package main

import (
	"fmt"
	"runtime"
)

func main() {
	fmt.Println("golang version:", runtime.Version())
	defer_demo()
}

func defer_demo() {
	for i := 0; i < 3; i++ {
		defer func() {
			fmt.Println(i)
		}()
	}
}

在1.21的版本中的结果:

复制代码
go run demo/defer2.go
golang version: go1.21.5
3
3
3

可以看到ids这个slice中的值都是10

在1.22的版本中的结果

复制代码
go run demo/defer2.go
golang version: go1.22.1
2
1
0

修改方法

第一个例子本意应该是让counter自增的,可以把counter的定义放在循环外面.修改后的代码

go 复制代码
package main

import (
	"fmt"
	"runtime"
)

func main() {
	fmt.Println("golang version:", runtime.Version())
	defer_demo()
}

func defer_demo() {
	counter := 0
	defer println()
	for n := 0; n < 3; n++ {
		defer func(v int) {
			fmt.Print(counter)
			counter++
		}(n)
	}
}

1.21和1.22的执行结果

复制代码
go run demo/defer.go 
golang version: go1.21.5
012

go run demo/defer.go
golang version: go1.22.1
012

针对第二个例子改动比较简单,把循环变量作为defer func的参数即可,修改后的代码

go 复制代码
package main

import (
	"fmt"
	"runtime"
)

func main() {
	fmt.Println("golang version:", runtime.Version())
	defer_demo()
}

func defer_demo() {
	for i := 0; i < 3; i++ {
		defer func(i int) {
			fmt.Println(i)
		}(i)
	}
}

这样1.21和1.22的结果就都一样了

复制代码
go run demo/defer2.go
golang version: go1.21.5
2
1
0


go run demo/defer2.go
golang version: go1.22.1
2
1
0

第二个例子如果把defer func换成一个closures也会出现1.21和1.22打印不一样的情况

代码

go 复制代码
package main

import (
	"fmt"
	"runtime"
)

func main() {
	fmt.Println("golang version:", runtime.Version())
	defer_demo()
}

func defer_demo() {
	var demoFunc func()
	for i := 0; i < 3; i++ {
		if demoFunc == nil {
			demoFunc = func() {
				fmt.Println(i)
			}
		}
		demoFunc()
	}
}

运行结果

复制代码
go run demo/closures.go
golang version: go1.21.5
0
1
2

go run demo/closures.go
golang version: go1.22.1
0
0
0
相关推荐
BD_Marathon24 分钟前
设计模式——依赖倒转原则
java·开发语言·设计模式
devmoon43 分钟前
在 Polkadot Runtime 中添加多个 Pallet 实例实战指南
java·开发语言·数据库·web3·区块链·波卡
Evand J1 小时前
TDOA(到达时间差)的GDOP和CRLB计算的MATLAB例程,论文复现,附参考文献。GDOP:几何精度因子&CRLB:克拉美罗下界
开发语言·matlab·tdoa·crlb·gdop
野犬寒鸦1 小时前
从零起步学习并发编程 || 第七章:ThreadLocal深层解析及常见问题解决方案
java·服务器·开发语言·jvm·后端·学习
云姜.1 小时前
java抽象类和接口
java·开发语言
xyq20241 小时前
Pandas 安装指南
开发语言
xixixin_1 小时前
【JavaScript 】从 || 到??:JavaScript 空值处理的最佳实践升级
开发语言·javascript·ecmascript
m0_736919101 小时前
C++中的委托构造函数
开发语言·c++·算法
lsx2024062 小时前
Python3 SMTP发送邮件教程
开发语言
懈尘2 小时前
从 Java 1.7 到 Java 21:逐版本深入解析新特性与平台演进
java·开发语言