目录
Mutex 和 RWMutex
WaitGroup
Cond
Once
Conclusion
首页 后端开发 Golang Go 语言中的并发安全怎样实现?

Go 语言中的并发安全怎样实现?

Jun 10, 2023 am 08:13 AM
go语言 实现 并发安全

随着计算机技术的不断发展,我们必须从单线程转向多线程来针对程序的处理。相对于传统并发处理模式,Go语言强大的并发处理机制吸引了众多开发者的注意。Go语言提供了一种轻量级的实现机制,使得地道的并发代码可以更加容易地编写。

然而,不可避免的是,多线程环境会带来很多竞争条件(Race Condition)。当多个线程尝试同时读写同一个共享资源时,由于执行的顺序的不确定性,可能会出现意外的结果。Race Condition 是开发者最害怕的潜在问题之一。

为了避免并发处理中出现潜在的问题,Go语言提供了种类丰富的标准库:sync。本文将介绍通过 sync 库实现并发安全的机制。

Mutex 和 RWMutex

mutex 是最常用的机制。在任意时刻,只能有一个协程能够获得 mutex 对象,而其他协程需要等待前一个协程释放锁之后才能够继续执行。Mutex 可以用于保护共享资源,代码安全而稳定地运行。

RWMutex 是另一种互斥锁类型,相当于 mutex 在读写领域的扩展。RWMutex 包含了两个计数器:读计数器和写计数器。

  • 当读协程还在进行读操作时,写操作会被锁住,并等待读操作结束。
  • 当写协程在调用锁操作时,所有正在进行的协程的读写操作都会被锁住。

这种机制保证同时有多个协程能够进行读操作,并且只有单个协程能够进行写操作。

var rwMutex sync.RWMutex
var count int
func read() {
    rwMutex.RLock()
    defer rwMutex.RUnlock()

    fmt.Println(count)
}

func write() {
    rwMutex.Lock()
    defer rwMutex.Unlock()

    count++
}
登录后复制

在上述示例代码中,我们使用了 RWMutex 类型的锁来保护 count 变量的读写操作。当一个线程调用 write() 函数时,将会锁定写计数器,并阻止所有其它协程进行读写操作。而当一个线程调用 read() 函数时,将会锁定读计数器,并允许其它协程进行读操作。

WaitGroup

WaitGroup 用来等待一组协程执行完成。假设我们有 n 个协程需要执行,那么在主协程中,我们需要调用 waitGroup.Add(n)。在每个协程执行完成后调用 waitGroup.Done()。

func main() {
    var wg sync.WaitGroup

    for i := 0; i < 5; i++ {
        wg.Add(1)
        go func(n int) {
            fmt.Println("goroutine ", n)
            wg.Done()
        }(i)
    }
    wg.Wait()
}
登录后复制

在这个例子中,我们使用 WaitGroup 来等待每个 goroutine 的执行,最后等待所有 goroutine 完成之后,才结束 main 的执行过程。

Cond

当多个协程需要停止或执行某些特定操作时,我们可以使用 Cond。Cond 与锁和 WaitGroup 的结合使用是很常见的。它允许 goroutine 同时阻塞,直到一个条件变量发生变化。

var cond = sync.NewCond(&sync.RWMutex{})

func printOddNumbers() {
    for i := 0; i < 10; i++ {
        cond.L.Lock()
        if i%2 == 1 {
            fmt.Println(i)
            cond.Signal()
        } else {
            cond.Wait()
        }
        cond.L.Unlock()
    }
}

func printEvenNumbers() {
    for i := 0; i < 10; i++ {
        cond.L.Lock()
        if i%2 == 0 {
            fmt.Println(i)
            cond.Signal()
        } else {
            cond.Wait()
        }
        cond.L.Unlock()
    }
}
登录后复制

在上述代码示例中,我们使用了 Cond 来保证偶数和奇数的分别输出。每个协程都使用 sync.Mutex 来锁定 goroutine,并等待另一个协程首先访问共享变量,然后再监测获得变量的值。

Once

在某些情况下,需要确保只执行一次某些操作,例如只读取一次配置文件或者只初始化一次全局状态。Go语言的 sync.Once 类型正是为此而生。当第一次调用函数时它会执行其内部的代码,之后调用都不会再次执行。

var once sync.Once

func doSomething() {
    once.Do(func() {
        fmt.Println("Do something")
    })
}
登录后复制

在上述示例中,我们使用 sync.Once 安全地执行了 doSomething 函数。在第一次调用 doSomething 时,函数会使用 once.Do() 只执行一次。

Conclusion

在本篇文章中,我们介绍了 Go 语言中常用的锁和机制,以保证并发代码的安全。使用 sync 库的 Mutex、RWMutex、WaitGroup、Cond 和 Once 类型都非常强大,可用于设计安全且高效的并发程序。随着并发机制的不断发展,了解并发编程的最新进展是保持开发技能竞争力的关键。

以上是Go 语言中的并发安全怎样实现?的详细内容。更多信息请关注PHP中文网其他相关文章!

本站声明
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智能驱动的应用程序,用于创建逼真的裸体照片

AI Clothes Remover

AI Clothes Remover

用于从照片中去除衣服的在线人工智能工具。

Undress AI Tool

Undress AI Tool

免费脱衣服图片

Clothoff.io

Clothoff.io

AI脱衣机

AI Hentai Generator

AI Hentai Generator

免费生成ai无尽的。

热门文章

R.E.P.O.能量晶体解释及其做什么(黄色晶体)
1 个月前 By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O.最佳图形设置
1 个月前 By 尊渡假赌尊渡假赌尊渡假赌
威尔R.E.P.O.有交叉游戏吗?
1 个月前 By 尊渡假赌尊渡假赌尊渡假赌

热工具

记事本++7.3.1

记事本++7.3.1

好用且免费的代码编辑器

SublimeText3汉化版

SublimeText3汉化版

中文版,非常好用

禅工作室 13.0.1

禅工作室 13.0.1

功能强大的PHP集成开发环境

Dreamweaver CS6

Dreamweaver CS6

视觉化网页开发工具

SublimeText3 Mac版

SublimeText3 Mac版

神级代码编辑软件(SublimeText3)

Go语言中用于浮点数运算的库有哪些? Go语言中用于浮点数运算的库有哪些? Apr 02, 2025 pm 02:06 PM

Go语言中用于浮点数运算的库介绍在Go语言(也称为Golang)中,进行浮点数的加减乘除运算时,如何确保精度是�...

Go的爬虫Colly中Queue线程的问题是什么? Go的爬虫Colly中Queue线程的问题是什么? Apr 02, 2025 pm 02:09 PM

Go爬虫Colly中的Queue线程问题探讨在使用Go语言的Colly爬虫库时,开发者常常会遇到关于线程和请求队列的问题。�...

在 Go 语言中,为什么使用 Println 和 string() 函数打印字符串会出现不同的效果? 在 Go 语言中,为什么使用 Println 和 string() 函数打印字符串会出现不同的效果? Apr 02, 2025 pm 02:03 PM

Go语言中字符串打印的区别:使用Println与string()函数的效果差异在Go...

在Go语言中使用Redis Stream实现消息队列时,如何解决user_id类型转换问题? 在Go语言中使用Redis Stream实现消息队列时,如何解决user_id类型转换问题? Apr 02, 2025 pm 04:54 PM

Go语言中使用RedisStream实现消息队列时类型转换问题在使用Go语言与Redis...

GoLand中自定义结构体标签不显示怎么办? GoLand中自定义结构体标签不显示怎么办? Apr 02, 2025 pm 05:09 PM

GoLand中自定义结构体标签不显示怎么办?在使用GoLand进行Go语言开发时,很多开发者会遇到自定义结构体标签在�...

Go语言中哪些库是由大公司开发或知名的开源项目提供的? Go语言中哪些库是由大公司开发或知名的开源项目提供的? Apr 02, 2025 pm 04:12 PM

Go语言中哪些库是大公司开发或知名开源项目?在使用Go语言进行编程时,开发者常常会遇到一些常见的需求,�...

Go语言中`var`和`type`关键字定义结构体的区别是什么? Go语言中`var`和`type`关键字定义结构体的区别是什么? Apr 02, 2025 pm 12:57 PM

Go语言中结构体定义的两种方式:var与type关键字的差异Go语言在定义结构体时,经常会看到两种不同的写法:一�...

使用 sql.Open 时,DSN 传空为什么不报错? 使用 sql.Open 时,DSN 传空为什么不报错? Apr 02, 2025 pm 12:54 PM

使用sql.Open时,DSN传空为什么不报错?在Go语言中,sql.Open...

See all articles