Golang 并发机制-2:Golang Goroutine 和竞争条件

devtools/2025/2/3 1:06:40/

在今天的软件开发中,我们正在使用并发的概念,它允许一次执行多个任务。在Go编程中,理解Go例程是至关重要的。本文试图详细解释什么是例程,它们有多轻,通过简单地使用“go”关键字创建它们,以及可能出现的竞争条件和共享数据问题等主要同步困难。

Goroutine 介绍

Goroutine 是 Go 编程语言中并发编程的基本构建块。它本质上是一个轻量级的执行线程,在 Go 程序中与其他 Goroutine 并行运行。与其它编程语言中的传统线程不同,Goroutine 由 Go 运行时管理,在内存和 CPU 利用率方面都更高效。

  • 性质与效率

Goroutine的突出特性之一是它们的轻量级特性。传统线程可能是资源密集型的,会消耗大量的内存和CPU资源。相比之下,gooutine非常高效,支持创建数千个这样的例程,而不会造成显著的开销。

Goroutine的效率源于它们能够跨较少数量的OS线程进行多路复用,并根据工作负载动态调整其分配。这意味着Go程序可以有效地利用多个内核和处理器,而不需要大量的手动线程管理。
在这里插入图片描述

Goroutine示例(‘ go ’关键字)

在Go中创建Goroutine非常简单,这要归功于“Go”关键字。当你在函数调用前加上‘ go ’时,go会创建新的gooutine来并发地执行该函数。

import ("fmt""time"
)func sayHello() {for i := 0; i < 5; i++ {fmt.Println("Hello, World!")time.Sleep(time.Millisecond * 500)}
}func main() {go sayHello() // Start a new Goroutinetime.Sleep(time.Second * 2)fmt.Println("Main function")
}

在上面的例子中,‘ sayHello ’函数与‘ main ’函数并发执行,使其成为在Go中利用并发性的一种简单而有效的方法。

竞争条件

虽然在并发编程中提供了许多优点,但它们也带来了必须小心管理的同步挑战

什么是竞态条件?

当多个Goroutine(轻量级线程)并发地访问共享数据,并且其中至少有一个需要修改数据时,就会出现竞争条件。竞争条件会导致不可预测的结果,因为执行顺序无法保证。它们可能导致数据损坏、崩溃或错误的程序行为。

竞态条件示例:

package mainimport ("fmt""sync"
)var sharedCounter int
var wg sync.WaitGroupfunc increment() {for i := 0; i < 10000; i++ {sharedCounter++}wg.Done()
}func main() {wg.Add(2)go increment()go increment()wg.Wait()fmt.Println("Shared Counter:", sharedCounter)
}

在本例中,两个goroutine在没有同步的情况下并发地增加‘ sharedCounter ’变量。这可能导致竞争条件,其中‘ sharedCounter ’的最终值是不可预测的,并且可能是不正确的。

假设 sharedCounter 的初始值为 0,两个 goroutine 同时读取到 0,然后分别加 1 并写回内存,最终 sharedCounter 的值会是 1 而不是 2。为了避免竞争条件,可以使用同步机制,如互斥锁(sync.Mutex)来保护对共享资源的访问。

规避竞争条件

为了消除Go中的竞争条件,你可以使用同步原语,例如互斥锁(互斥锁的简称)。互斥锁确保一次只有一个程序可以访问代码的关键部分。下面是上一个例子的更新版本,使用互斥锁进行正确的同步:

package mainimport ("fmt""sync"
)var sharedCounter int
var wg sync.WaitGroup
var mu sync.Mutexfunc increment() {for i := 0; i < 10000; i++ {mu.Lock()sharedCounter++mu.Unlock()}wg.Done()
}func main() {wg.Add(2)go increment()go increment()wg.Wait()fmt.Println("Shared Counter:", sharedCounter)
}

在修改后的代码中,我们使用‘ mu ’互斥锁来保护‘ sharedCounter ’被修改的代码的临界区。通过锁定和解锁互斥锁,我们确保一次只有一个Goroutine可以访问和修改‘ sharedCounter ’,从而消除了竞争条件。

理解共享数据问题

Go中的共享数据问题发生在多个线程在没有适当同步的情况下并发访问和操作共享数据时。这些问题主要表现为两种形式:

  1. 数据竞争:当两个或多个例程同时访问共享数据时,就会发生数据竞争,从而导致不可预测的结果。数据竞争可能导致数据损坏或不正确的程序行为。

  2. 死锁:当线程被卡住时,会发生死锁,等待彼此释放资源。这可能导致程序陷入停滞。

为了缓解Go中的共享数据问题,开发人员应该使用适当的同步机制,如互斥锁、通道和其他同步原语。以下是一些最佳做法:

  • 使用互斥锁:使用互斥锁保护共享数据,以确保一次只有一个线程可以访问它。
  • 使用通道:通道为程序提供了一种安全的方式来通信和共享数据。它们通过确保对共享数据的受控访问来防止数据竞争。
  • 避免循环依赖:在创建循环依赖时要小心,因为在循环依赖中,各例程会等待彼此释放资源,从而导致死锁。

总之,在用Go编写并发程序时,管理竞争条件和共享数据问题是至关重要的。通过理解这些问题并实现适当的同步技术,开发人员可以创建健壮可靠的并发应用程序,充分利用Go的并发支持,同时避免与共享数据操作相关的陷阱。

最后总结

总之,Go程序是Go编程语言的一个强大功能,它提供了一种轻量级且高效的方式来实现并发性。通过使用“go”关键字,开发人员可以轻松地创建并发执行任务的goroutine。然而,在用Go构建并发应用程序时,必须意识到同步挑战,比如竞争条件和共享数据问题,并采用适当的技术来解决这些问题。


http://www.ppmy.cn/devtools/155582.html

相关文章

人工智能导论--第1章-知识点与学习笔记

请根据教材内容&#xff0c;完成进行下面的作业任务。必须包含有教材的具体内容&#xff0c;不能是生成式AI系统的生成内容。 参考教材1.1节的内容介绍&#xff0c;谈谈你对“智能”的认识。思维能力是智能的重要特征之一&#xff0c;结合教材1.1.2节内容&#xff0c;从思维的…

国内优秀的FPGA设计公司主要分布在哪些城市?

近年来&#xff0c;国内FPGA行业发展迅速&#xff0c;随着5G通信、人工智能、大数据等新兴技术的崛起&#xff0c;FPGA设计企业的需求也迎来了爆发式增长。很多技术人才在求职时都会考虑城市的行业分布和发展潜力。因此&#xff0c;国内优秀的FPGA设计公司主要分布在哪些城市&a…

92,[8] 攻防世界 web Web_php_wrong_nginx_config

进入靶场 admin 123 还尝试了很多&#xff0c;都是建设中 进行目录扫描&#xff0c;扫描到了admin和robots.txt 这句话就应该想到BP 抓包 这样修改请求即可登录 BP加载不出来 F12修改 管理中心点击后会发生url的改变 变成这样 目录穿越&#xff0c;不断尝试 /admin/admi…

git中有关old mode 100644、new mode 10075的问题解决小结

在 Git 版本控制系统中&#xff0c;文件权限变更是一种常见情况。当你看到类似 old mode 100644 和 new mode 100755 的信息时&#xff0c;这通常表示文件的权限发生了变化。本文将详细解析这种情况&#xff0c;并提供解决方法和注意事项。 问题背景 在 Git 中&#xff0c;文…

Hot100之哈希

1两数之和 题目 思路解析 解法1--两次循环 解法2--哈希表一次循环 代码 解法1--两次循环 class Solution {public int[] twoSum(int[] nums, int target) {int nums1[] new int[2];int length nums.length;for (int i 0; i < length; i) {for (int j i 1; j < …

STM32-CAN总线

1.CAN总线简介 CAN总线是由BOSCH公司开发的一种简洁易用、传输速度快、易扩展、可靠性高的串行通信总线 2.CAN总线特征 两根通信线&#xff08;CAN_H、CAN_L&#xff09;&#xff0c;线路少&#xff0c;无需共地差分信号通信&#xff08;相对的是单端信号&#xff09;&#…

1、云计算

云是一种基于互联网的计算技术和服务模式&#xff0c;它可以将计算资源、存储资源、软件资源等进行整合和虚拟化&#xff0c;以按需使用、可灵活扩展的方式提供给用户&#xff0c;就像把传统的本地计算资源和服务放到了一个庞大的 “云端”&#xff0c;用户可以通过网络随时随地…

Ubuntu16.04编译安装Cartographer 1.0版本

说明 官方文档 由于Ubuntu16.04已经是很老的系统&#xff0c;如果直接按照Cartographer官方安装文档安装会出现代码编译失败的问题&#xff0c;本文给出了解决这些问题的办法。正常情况下执行本文给出的安装方法即可成功安装。 依赖安装 # 这里和官方一致 # Install the req…