Go语言竞争状态简述
来源:云海天教程
时间:2022-12-28 08:26:21 222浏览 收藏
对于一个Golang开发者来说,牢固扎实的基础是十分重要的,golang学习网就来带大家一点点的掌握基础知识点。今天本篇文章带大家了解《Go语言竞争状态简述》,主要介绍了并发,希望对大家的知识积累有所帮助,快点收藏起来吧,否则需要时就找不到了!
Go语言中如果两个或者多个 goroutine 在没有互相同步的情况下,访问某个共享的资源,并试图同时读和写这个资源,就处于相互竞争的状态,这种情况被称作竞争状态(race candition)。竞争状态的存在是让并发程序变得复杂的地方,十分容易引起潜在问题。对一个共享资源的读和写操作必须是原子化的,换句话说,同一时刻只能有一个 goroutine 对共享资源进行读和写操作。
【示例】包含竞争状态的示例程序。
// 这个示例程序展示如何在程序里造成竞争状态// 实际上不希望出现这种情况package mainimport ( "fmt" "runtime" "sync")var ( // counter 是所有goroutine 都要增加其值的变量 counter int // wg 用来等待程序结束 wg sync.WaitGroup)// main 是所有Go 程序的入口func main() { // 计数加 2,表示要等待两个goroutine wg.Add(2) // 创建两个goroutine go incCounter(1) go incCounter(2) // 等待 goroutine 结束 wg.Wait() fmt.Println("Final Counter:", counter)}// incCounter 增加包里counter 变量的值func incCounter(id int) { // 在函数退出时调用Done 来通知main 函数工作已经完成 defer wg.Done() for count := 0; count 输出结果如下所示。
Final Counter: 2
变量 counter 会进行 4 次读和写操作,每个 goroutine 执行两次。但是,程序终止时,counter 变量的值为 2。下图提供了为什么会这样的线索。图:竞争状态下程序行为的图像表达
每个 goroutine 都会覆盖另一个 goroutine 的工作。这种覆盖发生在goroutine 切换的时候。每个 goroutine 创造了一个 counter 变量的副本,之后就切换到另一个 goroutine。当这个 goroutine 再次运行的时候,counter 变量的值已经改变了,但是 goroutine 并没有更新自己的那个副本的值,而是继续使用这个副本的值,用这个值递增,并存回 counter 变量,结果覆盖了另一个 goroutine 完成的工作。
代码说明如下:
在第 25 行和第 26 行,使用 incCounter 函数创建了两个 goroutine。在第 34 行,incCounter 函数对包内变量 counter 进行了读和写操作,而这个变量是这个示例程序里的共享资源。每个 goroutine 都会先读出这个 counter 变量的值。在第 40 行将 counter 变量的副本存入一个叫作 value 的本地变量。在第 46 行,incCounter 函数对 value 的副本的值加 1。在第 49 行将这个新值存回到 counter 变量。
这个函数在第 43 行调用了 runtime 包的 Gosched 函数,用于将 goroutine 从当前线程退出,给其他 goroutine 运行的机会。在两次操作中间这样做的目的是强制调度器切换两个 goroutine,以便让竞争状态的效果变得更明显。
Go语言有一个特别的工具,可以在代码里检测竞争状态。在查找这类错误的时候,这个工具非常好用,尤其是在竞争状态并不像这个例子里这么明显的时候。让我们用这个竞争检测器来检测一下我们的示例代码,代码如下所示。
go build -race // 用竞争检测器标志来编译程序./example // 运行程序==================WARNING: DATA RACEWrite by goroutine 5: main.incCounter() /example/main.go:49 +0x96Previous read by goroutine 6: main.incCounter() /example/main.go:40 +0x66Goroutine 5 (running) created at: main.main() /example/main.go:25 +0x5cGoroutine 6 (running) created at: main.main() /example/main.go:26 +0x73==================Final Counter: 2Found 1 data race(s)上述代码中的竞争检测器指出了 4 行代码有问题,如下所示。
Line 49: counter = value
Line 40: value := counter
Line 25: go incCounter(1)
Line 26: go incCounter(2)
以上就是本文的全部内容了,是否有顺利帮助你解决问题?若是能给你带来学习上的帮助,请大家多多支持golang学习网!更多关于Golang的相关知识,也可关注golang学习网公众号。
-
417 收藏
-
328 收藏
-
367 收藏
-
221 收藏
-
109 收藏
-
438 收藏
-
280 收藏
-
181 收藏
-
371 收藏
-
236 收藏
-
416 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 542次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 507次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 497次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 484次学习
-
- 健壮的丝袜
- 好细啊,已收藏,感谢老哥的这篇技术文章,我会继续支持!
- 2023-02-28 08:07:21
-
- 淡淡的万宝路
- 这篇博文出现的刚刚好,太详细了,太给力了,已收藏,关注楼主了!希望楼主能多写Golang相关的文章。
- 2023-02-22 18:33:34
-
- 无奈的御姐
- 这篇技术文章真是及时雨啊,楼主加油!
- 2023-02-19 16:15:45
-
- 欢喜的钢铁侠
- 很好,一直没懂这个问题,但其实工作中常常有遇到...不过今天到这,看完之后很有帮助,总算是懂了,感谢大佬分享技术文章!
- 2023-01-12 19:23:39
-
- 花痴的香烟
- 很有用,一直没懂这个问题,但其实工作中常常有遇到...不过今天到这,帮助很大,总算是懂了,感谢作者分享博文!
- 2022-12-30 06:46:32
-
- 微笑的歌曲
- 这篇技术贴出现的刚刚好,细节满满,真优秀,已加入收藏夹了,关注博主了!希望博主能多写Golang相关的文章。
- 2022-12-29 23:18:48