Go 为什么不支持可重入锁原理解析

 更新时间:2023年08月07日 09:02:49   作者:煎鱼  
这篇文章主要为大家介绍了Go 为什么不支持可重入锁原理解析,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪

引言

程序里的锁,是很多小伙伴在写分布式应用时用的最多的一个利器之一。

使用 Go 的同学里,绝大部分都有其他语言的经验,就会对其中一点有疑惑,那就是 Go 里的锁,竟然不支持可重入

为此,今天煎鱼带大家一起来了解这里的设计考量,看看为什么。

可重入锁

如果对已经上锁的普通互斥锁进行 “加锁” 操作,其结果要么失败,要么会阻塞至解锁。

锁的场景如下:

  • 在加锁上:如果是可重入互斥锁,当前尝试加锁的线程如果就是持有该锁的线程时,加锁操作就会成功。
  • 在解锁上:可重入互斥锁一般都会记录被加锁的次数,只有执行相同次数的解锁操作才会真正解锁。

简单来讲,可重入互斥锁是互斥锁的一种,同一线程对其多次加锁不会产生死锁,又或是导致阻塞。

不同语言间实现可能或多或少有些区别,但大体意思差不多。

请你想一下,Go 是怎么样的呢?

Go 支持情况

我们看到以下这个 Go 互斥锁例子:

var mu sync.Mutex
func main() {
    mu.Lock()
    mu.Lock()
}

这段 Go 程序会阻塞吗?不会,会报以下错误:

fatal error: all goroutines are asleep - deadlock!

Go 显然是不支持可重入互斥锁的。

官方回复

Go 设计原则

在工程中使用互斥的根本原因是:为了保护不变量,也可以用于保护内、外部的不变量。

基于此,Go 在互斥锁设计上会遵守这几个原则。如下:

  • 在调用 mutex.Lock 方法时,要保证这些变量的不变性保持,不会在后续的过程中被破坏。
  • 在调用 mu.Unlock 方法时,要保证:

    • 程序不再需要依赖那些不变量。
    • 如果程序在互斥锁加锁期间破坏了它们,则需要确保已经恢复了它们。

不支持的原因

讲了 Go 自己的设计原则后,那为什么不支持可重入呢?

其实 Russ Cox 于 2010 年在《Experimenting with GO》就给出了答复,认为递归(又称:重入)互斥是个坏主意,这个设计并不好。

我们可以结合官方的例子来理解。

如下:

func F() {
        mu.Lock()
        ... do some stuff ...
        G()
        ... do some more stuff ...
        mu.Unlock()
}
func G() {
        mu.Lock()
        ... do some stuff ...
        mu.Unlock()
}

在上述代码中,我们在 F 方法中调用 mu.Lock 方法加上了锁。如果支持可重入锁,接着就会进入到 G 方法中。

此时就会有一个致命的问题,你不知道 F 和 G 方法加锁后是不是做了什么事情,从而导致破坏了不变量,毕竟随手起几个协程做点坏事,也是完全可能的。

这对于 Go 是无法接受的,可重入的设计违反了前面所提到的设计理念,也就是:“要保证这些变量的不变性保持,不会在后续的过程中被破坏”。

基于上述原因,Go 官方团队选择了没有支持该项特性。

总结

Go 互斥锁没有支持可重入锁的设计,也是喜欢的大道至简的思路了,可能的干扰比较多,不如直接简单的来。

以上就是Go 为什么不支持可重入锁原理解析的详细内容,更多关于Go不支持可重入锁的资料请关注脚本之家其它相关文章!

相关文章

  • Go语言如何实现Benchmark函数

    Go语言如何实现Benchmark函数

    go想要在main函数中测试benchmark会麻烦一些,所以这篇文章主要为大家介绍了如何实现了一个简单的且没有开销的benchmark函数,希望对大家有所帮助
    2024-12-12
  • Golang中的内存泄漏你真的理解了吗

    Golang中的内存泄漏你真的理解了吗

    内存泄漏是编程中常见的问题,会对程序的性能和稳定性产生严重影响,本文将深入详解 Golang 中的内存泄漏的原因、检测方法以及避免方法,希望对大家有所帮助
    2023-12-12
  • go语言字符串的拼接和切片方法总结

    go语言字符串的拼接和切片方法总结

    在go语言中,因为字符串只能被访问,不能被修改,所以进行字符串拼接的时候,golang都需要进行内存拷贝,造成一定的性能消耗,这篇文章主要给大家介绍了关于go语言字符串的拼接和切片的相关资料,需要的朋友可以参考下
    2022-11-11
  • Go错误处理之panic函数和recover函数使用及捕获异常方法

    Go错误处理之panic函数和recover函数使用及捕获异常方法

    这篇文章主要介绍了Go错误处理之panic函数使用及捕获,本篇探讨了如何使用 panic 和 recover 来处理 Go 语言中的异常,需要的朋友可以参考下
    2023-03-03
  • GoLand安装与环境配置的完整步骤

    GoLand安装与环境配置的完整步骤

    作为一个go语言程序员,觉得自己有义务为go新手开一条更简单便捷的上手之路,下面这篇文章主要给大家介绍了关于GoLand安装与环境配置的完整步骤,文中通过图文介绍的非常详细,需要的朋友可以参考下
    2022-12-12
  • go内存缓存BigCache之Entry封装源码阅读

    go内存缓存BigCache之Entry封装源码阅读

    这篇文章主要介绍了go内存缓存BigCache之Entry封装源码阅读
    2023-09-09
  • Golang WaitGroup实现原理解析

    Golang WaitGroup实现原理解析

    WaitGroup是Golang并发的两种方式之一,一个是Channel,另一个是WaitGroup,下面这篇文章主要给大家介绍了关于golang基础之waitgroup用法以及使用要点的相关资料,需要的朋友可以参考下
    2023-02-02
  • Golang中关于defer的盲区梳理

    Golang中关于defer的盲区梳理

    关于Go中的defer,是做什么的?执行顺序是怎么样的?相信学过Go语言的同学,已经不在陌生,今天就来讲讲其中需要掌握的几个知识点,希望对大家有所帮助
    2023-03-03
  • go使用SQLX操作MySQL数据库的教程详解

    go使用SQLX操作MySQL数据库的教程详解

    sqlx 是 Go 语言中一个流行的操作数据库的第三方包,它提供了对 Go 标准库 database/sql 的扩展,简化了操作数据库的步骤,下面我们就来学习一下go如何使用SQLX实现MySQL数据库的一些基本操作吧
    2023-11-11
  • Goland激活码破解永久版及安装详细教程(亲测可以)

    Goland激活码破解永久版及安装详细教程(亲测可以)

    这篇文章主要介绍了Goland激活码破解永久版及安装详细教程(亲测可以),本文通过实例图文相结合给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2020-10-10

最新评论