loopvar 改动不同版本的影响-并发

2024-04-10 21:52

本文主要是介绍loopvar 改动不同版本的影响-并发,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

看一个关于并发的例子

package mainimport ("fmt""runtime""sync"
)func main() {fmt.Println("golang version:", runtime.Version())concurrencyDemo()
}func concurrencyDemo() {isGold := func(num uint64) bool {return num%65536 == 0}var c = make(chan uint64)var m sync.Mutexfor n, i := 0, uint64(0); n < 5; n++ {go func() {for {m.Lock()i++v := im.Unlock()if isGold(v) {c <- v}}}()}for n := range c {fmt.Println("Found gold:", n)}
}

这个代码定义了一个isGold 的func,可以被65536整除的就是gold,然后定义了一个uint64的channel,接着开了5个go routine,每个go routine都是一个不会停止的协程,加锁i进行自增,并把自增后的i复制给v,如果v可以被65536整除,那么把v放到uint64的channel中,最后打印出channel的值。
如果代码写的不好,并发的时候是很容易出现data race的,运行的时候加上-race这个参数

golang 1.21 运行结果

 go run -race demo/concurrency.go
golang version: go1.21.5
Found gold: 65536
Found gold: 131072
Found gold: 196608
Found gold: 262144
Found gold: 327680
Found gold: 393216
Found gold: 458752
Found gold: 524288
Found gold: 589824
Found gold: 655360
^Csignal: interrupt

golang 1.22运行结果

go run -race demo/concurrency.go
golang version: go1.22.1
==================
WARNING: DATA RACE
Write at 0x00c000014148 by goroutine 7:main.concurrencyDemo.func2()/home/lfeng/GolandProjects/golang101/demo/concurrency.go:24 +0x65Previous read at 0x00c000014148 by main goroutine:main.concurrencyDemo()/home/lfeng/GolandProjects/golang101/demo/concurrency.go:20 +0x87main.main()/home/lfeng/GolandProjects/golang101/demo/concurrency.go:11 +0xc4Goroutine 7 (running) created at:main.concurrencyDemo()/home/lfeng/GolandProjects/golang101/demo/concurrency.go:21 +0x7amain.main()/home/lfeng/GolandProjects/golang101/demo/concurrency.go:11 +0xc4
==================
Found gold: 65536
Found gold: 65536
Found gold: 65536
Found gold: 65536
Found gold: 131072
Found gold: 131072
Found gold: 131072
Found gold: 65536
^Csignal: interrupt

可以看到1.21是没有data race的,但是1.22存在了data race

解决方法

把i的定义移到循环外面,修改后的代码

package mainimport ("fmt""runtime""sync"
)func main() {fmt.Println("golang version:", runtime.Version())concurrencyDemo()
}func concurrencyDemo() {isGold := func(num uint64) bool {return num%65536 == 0}var c = make(chan uint64)var m sync.Mutexi := uint64(0)for n := 0; n < 5; n++ {go func() {for {m.Lock()i++v := im.Unlock()if isGold(v) {c <- v}}}()}for n := range c {fmt.Println("Found gold:", n)}
}

修改后的代码不会存在data race了

再看个例子

package mainimport ("fmt""runtime""sync"
)func main() {fmt.Println("golang version:", runtime.Version())concurrencyDemo()
}func concurrencyDemo() {var wg sync.WaitGroupwg.Add(5)for i := 0; i < 5; i++ {go func() {defer wg.Done()fmt.Println(i)}()}
}

这个例子其实很简单,使用waitGroup来控制协程的数量和协程的结束。协程中只是打印循环变量i

golang 1.21的运行结果

go run -race demo/concurrency2.go
golang version: go1.21.5
5
5
==================
WARNING: DATA RACE
Read at 0x00c000120028 by goroutine 9:main.concurrencyDemo.func1()/home/lfeng/GolandProjects/golang101/demo/concurrency2.go:20 +0xafPrevious write at 0x00c000120028 by main goroutine:main.concurrencyDemo()/home/lfeng/GolandProjects/golang101/demo/concurrency2.go:17 +0xa4main.main()/home/lfeng/GolandProjects/golang101/demo/concurrency2.go:11 +0xc4Goroutine 9 (running) created at:main.concurrencyDemo()/home/lfeng/GolandProjects/golang101/demo/concurrency2.go:18 +0x85main.main()/home/lfeng/GolandProjects/golang101/demo/concurrency2.go:11 +0xc4
==================
5
5
5
Found 1 data race(s)
exit status 66

golang 1.22运行结果

go run -race demo/concurrency2.go
golang version: go1.22.1
2
1
4
3
0

发现1.21存在data race,但是1.22是不存在data race的

解决方法

go func增加参数传递循环的i
修改后的代码

package mainimport ("fmt""runtime""sync"
)func main() {fmt.Println("golang version:", runtime.Version())concurrencyDemo()
}func concurrencyDemo() {var wg sync.WaitGroupwg.Add(5)for i := 0; i < 5; i++ {go func(i int) {defer wg.Done()fmt.Println(i)}(i)}
}

修改后的代码也不会存在data race

结论

上面两个例子出现data race的原因都是1.21 loop 变量是初始化一次,但是1.22每次循环都会创建新变量

这篇关于loopvar 改动不同版本的影响-并发的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/892248

相关文章

SpringBoot实现不同接口指定上传文件大小的具体步骤

《SpringBoot实现不同接口指定上传文件大小的具体步骤》:本文主要介绍在SpringBoot中通过自定义注解、AOP拦截和配置文件实现不同接口上传文件大小限制的方法,强调需设置全局阈值远大于... 目录一  springboot实现不同接口指定文件大小1.1 思路说明1.2 工程启动说明二 具体实施2

sysmain服务可以禁用吗? 电脑sysmain服务关闭后的影响与操作指南

《sysmain服务可以禁用吗?电脑sysmain服务关闭后的影响与操作指南》在Windows系统中,SysMain服务(原名Superfetch)作为一个旨在提升系统性能的关键组件,一直备受用户关... 在使用 Windows 系统时,有时候真有点像在「开盲盒」。全新安装系统后的「默认设置」,往往并不尽编

Web服务器-Nginx-高并发问题

《Web服务器-Nginx-高并发问题》Nginx通过事件驱动、I/O多路复用和异步非阻塞技术高效处理高并发,结合动静分离和限流策略,提升性能与稳定性... 目录前言一、架构1. 原生多进程架构2. 事件驱动模型3. IO多路复用4. 异步非阻塞 I/O5. Nginx高并发配置实战二、动静分离1. 职责2

Ubuntu如何升级Python版本

《Ubuntu如何升级Python版本》Ubuntu22.04Docker中,安装Python3.11后,使用update-alternatives设置为默认版本,最后用python3-V验证... 目China编程录问题描述前提环境解决方法总结问题描述Ubuntu22.04系统自带python3.10,想升级

Spring Security 前后端分离场景下的会话并发管理

《SpringSecurity前后端分离场景下的会话并发管理》本文介绍了在前后端分离架构下实现SpringSecurity会话并发管理的问题,传统Web开发中只需简单配置sessionManage... 目录背景分析传统 web 开发中的 sessionManagement 入口ConcurrentSess

Python Flask实现定时任务的不同方法详解

《PythonFlask实现定时任务的不同方法详解》在Flask中实现定时任务,最常用的方法是使用APScheduler库,本文将提供一个完整的解决方案,有需要的小伙伴可以跟随小编一起学习一下... 目录完js整实现方案代码解释1. 依赖安装2. 核心组件3. 任务类型4. 任务管理5. 持久化存储生产环境

MySQL中处理数据的并发一致性的实现示例

《MySQL中处理数据的并发一致性的实现示例》在MySQL中处理数据的并发一致性是确保多个用户或应用程序同时访问和修改数据库时,不会导致数据冲突、数据丢失或数据不一致,MySQL通过事务和锁机制来管理... 目录一、事务(Transactions)1. 事务控制语句二、锁(Locks)1. 锁类型2. 锁粒

更改linux系统的默认Python版本方式

《更改linux系统的默认Python版本方式》通过删除原Python软链接并创建指向python3.6的新链接,可切换系统默认Python版本,需注意版本冲突、环境混乱及维护问题,建议使用pyenv... 目录更改系统的默认python版本软链接软链接的特点创建软链接的命令使用场景注意事项总结更改系统的默

Linux升级或者切换python版本实现方式

《Linux升级或者切换python版本实现方式》本文介绍在Ubuntu/Debian系统升级Python至3.11或更高版本的方法,通过查看版本列表并选择新版本进行全局修改,需注意自动与手动模式的选... 目录升级系统python版本 (适用于全局修改)对于Ubuntu/Debian系统安装后,验证Pyt

深入解析Java NIO在高并发场景下的性能优化实践指南

《深入解析JavaNIO在高并发场景下的性能优化实践指南》随着互联网业务不断演进,对高并发、低延时网络服务的需求日益增长,本文将深入解析JavaNIO在高并发场景下的性能优化方法,希望对大家有所帮助... 目录简介一、技术背景与应用场景二、核心原理深入分析2.1 Selector多路复用2.2 Buffer