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
相关推荐
花酒锄作田8 天前
Gin 框架中的规范响应格式设计与实现
golang·gin
郑州光合科技余经理8 天前
代码展示:PHP搭建海外版外卖系统源码解析
java·开发语言·前端·后端·系统架构·uni-app·php
feifeigo1238 天前
matlab画图工具
开发语言·matlab
dustcell.8 天前
haproxy七层代理
java·开发语言·前端
norlan_jame8 天前
C-PHY与D-PHY差异
c语言·开发语言
多恩Stone8 天前
【C++入门扫盲1】C++ 与 Python:类型、编译器/解释器与 CPU 的关系
开发语言·c++·人工智能·python·算法·3d·aigc
QQ4022054968 天前
Python+django+vue3预制菜半成品配菜平台
开发语言·python·django
遥遥江上月8 天前
Node.js + Stagehand + Python 部署
开发语言·python·node.js
m0_531237178 天前
C语言-数组练习进阶
c语言·开发语言·算法
Railshiqian8 天前
给android源码下的模拟器添加两个后排屏的修改
android·开发语言·javascript