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
相关推荐
鲤籽鲲2 分钟前
C# MethodTimer.Fody 使用详解
开发语言·c#·mfc
亚图跨际6 分钟前
Python和R荧光分光光度法
开发语言·python·r语言·荧光分光光度法
Rverdoser14 分钟前
RabbitMQ的基本概念和入门
开发语言·后端·ruby
dj244294570717 分钟前
JAVA中的Lamda表达式
java·开发语言
流星白龙1 小时前
【C++习题】10.反转字符串中的单词 lll
开发语言·c++
尘浮生1 小时前
Java项目实战II基于微信小程序的校运会管理系统(开发文档+数据库+源码)
java·开发语言·数据库·微信小程序·小程序·maven·intellij-idea
MessiGo1 小时前
Python 爬虫 (1)基础 | 基础操作
开发语言·python
Tech Synapse1 小时前
Java根据前端返回的字段名进行查询数据的方法
java·开发语言·后端
乌啼霜满天2491 小时前
JDBC编程---Java
java·开发语言·sql
色空大师1 小时前
23种设计模式
java·开发语言·设计模式