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
相关推荐
侃侃_天下1 天前
最终的信号类
开发语言·c++·算法
echoarts1 天前
Rayon Rust中的数据并行库入门教程
开发语言·其他·算法·rust
Aomnitrix1 天前
知识管理新范式——cpolar+Wiki.js打造企业级分布式知识库
开发语言·javascript·分布式
每天回答3个问题1 天前
UE5C++编译遇到MSB3073
开发语言·c++·ue5
伍哥的传说1 天前
Vite Plugin PWA – 零配置构建现代渐进式Web应用
开发语言·前端·javascript·web app·pwa·service worker·workbox
小莞尔1 天前
【51单片机】【protues仿真】 基于51单片机八路抢答器系统
c语言·开发语言·单片机·嵌入式硬件·51单片机
我是菜鸟0713号1 天前
Qt 中 OPC UA 通讯实战
开发语言·qt
JCBP_1 天前
QT(4)
开发语言·汇编·c++·qt·算法
Brookty1 天前
【JavaEE】线程安全-内存可见性、指令全排序
java·开发语言·后端·java-ee·线程安全·内存可见性·指令重排序
百锦再1 天前
[特殊字符] Python在CentOS系统执行深度指南
开发语言·python·plotly·django·centos·virtualenv·pygame