Golang 中的互斥锁是什么?

简介: #去#编程#初学者#建筑学

当您构建 Golang 程序时,您几乎总会发现 Goroutines 的用途。

Goroutines 功能强大且通常易于使用,但是,如果您需要修改 Goroutines 之间共享的某些数据,那么您可能会遇到数据完整性方面的问题。

在本文中,我们将了解什么是“互斥体”以及如何使用它。

什么是互斥体?

在Golang中;Goroutine 本质上是一个放在后台队列中并在资源可用时并发执行的函数。

package main
import (
    "fmt"
    "sync"
)
var NUM_PROCESSED = 0
func countNumProcessed(wg *sync.WaitGroup) {
    defer wg.Done()
    NUM_PROCESSED++
}
func main() {
    var wg sync.WaitGroup
    for i := 0; i < 500; i++ {
        wg.Add(1)
        go countNumProcessed(&wg)
    }
    wg.Wait()
    fmt.Println(NUM_PROCESSED)
}

如果我们删除“go”关键字,则对“countNumProcessed”的每个函数调用都会阻塞循环并等待该函数完成后再继续循环。

当您使用“go”关键字时,这些函数将同时运行。这使得 2 个或更多函数可以同时修改变量。

如果您运行此代码几次 - 您会注意到总计数会波动。

这是因为每个 Goroutine 在几纳秒内复制内存中的“NUM_PROCESSED”值以递增它,然后更新变量。

如果两个(或更多)Goroutines 大约在同一时间复制该值。例如:假设设置为“200”,那么每个都会添加“1”即“201”。因此,“NUM_PROCESSED”的值将变为“201”而不是“202”。

这就是互斥体派上用场的地方。互斥锁会在进程中创建一把“锁”——这样一次只有一个 Goroutine 可以更新“NUM_PROCESSED”。

然后其他 Goroutines 将暂停,直到锁被释放。这与队列非常相似 - 当每个 Goroutine 释放锁时,下一个 Goroutine 会获取新锁,并且该过程将继续,直到所有排队的 Goroutine 完成更新“NUM_PROCESSED”。

互斥体示例

我们可以修改上面的代码来引入一个Mutex,如下所示:

package main
import (
    "fmt"
    "sync"
)
var NUM_PROCESSED = 0
var MUTEX sync.Mutex
func countNumProcessed(wg *sync.WaitGroup) {
    defer wg.Done()
    MUTEX.Lock()
    NUM_PROCESSED++
    MUTEX.Unlock()
}
func main() {
    var wg sync.WaitGroup
    for i := 0; i < 500; i++ {
        wg.Add(1)
        go countNumProcessed(&wg)
    }
    wg.Wait()
    fmt.Println(NUM_PROCESSED)
}

您会注意到上面的代码,无论您运行这个函数多少次,它总是会打印“500”:这与 for 循环中创建的 goroutine 数量完全相同。

相关文章
GoLand软件包 home 包含多个 main 函数 请考虑改用文件种类
GoLand软件包 home 包含多个 main 函数 请考虑改用文件种类
336 0
GoLand软件包 home 包含多个 main 函数 请考虑改用文件种类
|
安全 测试技术 开发者
通义千问2.5有哪些升级
通义千问2.5有哪些升级
1341 5
|
数据采集 存储 编解码
一份简明的 Base64 原理解析
Base64 编码器的原理,其实很简单,花一点点时间学会它,你就又消除了一个知识盲点。
589 3
|
Java 关系型数据库 MySQL
基于Java的学生成绩管理系统/学生信息管理系统
基于Java的学生成绩管理系统/学生信息管理系统
316 2
|
消息中间件 监控 Java
接口请求重试策略:保障稳定性的必杀技
接口请求重试策略:保障稳定性的必杀技
1201 0
|
存储 固态存储 大数据
阿里云服务器实例、块存储、带宽收费标准与云服务器最新活动价格参考
阿里云服务器价格通常包括云服务器实例价格、块存储价格和带宽价格组成,云服务器不同实例规格收费标准不一样,选择不同类型的块存储收费标准也不一样,选择不同的带宽收费标准也不一样。现在阿里云轻量应用服务器2核4G4M峰值带宽298元1年,云服务器2核4G5M固定带宽199元1年、2核8G1M固定带宽652.32元1年、4核8G1M固定带宽955.58元1年、4核16G10M带宽100G ESSD Entry云盘70元1个月。本文为大家整理了目前阿里云服务器实例、块存储、带宽收费标准与云服务器最新的活动价格情况,以供参考。
阿里云服务器实例、块存储、带宽收费标准与云服务器最新活动价格参考
|
人工智能 运维 安全
【年终总结系列 2023】成长与收获:回顾过去、展望未来,加油2024!
【1月更文挑战第1天】年关将至,富余的时间也稍显多了些,遂写下此文,好好回顾一下自己这一年的收获,同时也立下2024年的新年flag。
|
C语言 定位技术 存储
【C语言基础入门】二级指针、一维数组与指针、二维数组与指针
【C语言基础入门】二级指针、一维数组与指针、二维数组与指针
442 0
【C语言基础入门】二级指针、一维数组与指针、二维数组与指针
|
JSON Linux C语言
全网最权威唯一值得推荐的《C/C++框架和库》
关于C++框架、库和资源的一些汇总列表,内容包括:标准库、Web应用框架、人工智能、数据库、图片处理、机器学习、日志、代码分析等。
566 1
|
算法 数据安全/隐私保护 数据格式
BUUCTF snake 1
BUUCTF snake 1
380 0