Go语言核心手册-9.互斥锁

简介: 本章的内容不多,主要需要注意互斥锁和读写锁的几条注意事项,读写锁其实就是更细粒度的锁划分,为了能让程序更好并发,上面已经讲述的非常清楚,这里就不再啰嗦。唯一再强调的一点,无论是互斥锁还是读写锁,我们都不要试图去解锁未锁定的锁,因为这样会引发不可恢复的 panic。

9.1 基础知识


对写操作的锁定和解锁,简称“写锁定”和“写解锁”:

func (*RWMutex)Lock() func (*RWMutex)Unlock()


对读操作的锁定和解锁,简称为“读锁定”与“读解锁”:

func (*RWMutex)RLock() func (*RWMutex)RUnlock()


看个不使用锁的示例:

func printer(str string) {   for _, data := range str {      fmt.Printf("%c", data)   }   fmt.Println()}func person1() {   printer("hello")}func person2() {   printer("world")}func main() {   go person1()   person2()time.Sleep(time.Second)}//输出结果//worhello//ld


加上互斥锁的示例:

var mut sync.Mutexfunc printer(str string) {   mut.Lock()   defer mut.Unlock()   for _, data := range str {      fmt.Printf("%c", data)   }   fmt.Println()}func person1() {   printer("hello")}func person2() {   printer("world")}func main() {   go person1()   person2()time.Sleep(time.Second)}//输出结果//world//hello


9.2 注意事项


9.2.1 互斥锁

  • 不要重复锁定互斥锁:对一个已经被锁定的互斥锁进行锁定,是会立即阻塞当前的goroutine,这个 goroutine所执行的流程,会一直停滞在调用该互斥锁的Lock方法的那行代码上。(注意:这种由 Go 语言运行时系统自行抛出的 panic 都属于致命错误,都是无法被恢复的,调用recover函数对它们起不到任何作用。也就是说,一旦产生死锁,程序必然崩溃。)
  • 不要忘记解锁互斥锁,必要时使用defer语句:因为在一个 goroutine 执行的流程中,可能会出现诸如“锁定、解锁、再锁定、再解锁”的操作,所以如果我们忘记了中间的解锁操作,那就一定会造成重复锁定。
var mutex sync.Mutex    func write() {     defer mutex.Unlock() // 通过defer解锁    mutex.Lock()     // 获取临界资源,执行具体逻辑... }
  • 不要对尚未锁定或者已解锁的互斥锁解锁:这个程序会直接panic。
var mutex sync.Mutex // 定义互斥锁变量 mutexmutex.Lock()mutex.Unlock()mutex.Unlock() // fatal error: sync: unlock of unlocked mutexreturn
  • 不要在多个函数之间直接传递互斥锁:互斥锁是一结构体类型,即值类型,把它传给一个函数、将它从函数中返回、把它赋给其他变量、让它进入某个通道都会导致它的副本的产生。因此,原值和它的副本、以及多个副本之间都是完全独立的,它们都是不同的互斥锁。


9.2.2 读写锁

  • 在写锁已被锁定的情况下再试图锁定写锁,会阻塞当前的 goroutine;
  • 在写锁已被锁定的情况下试图锁定读锁,也会阻塞当前的 goroutine;
  • 在读锁已被锁定的情况下试图锁定写锁,同样会阻塞当前的 goroutine;
  • 在读锁已被锁定的情况下再试图锁定读锁,并不会阻塞当前的 goroutine;
  • 解锁“读写锁中未被锁定的写锁”,会立即引发 panic,对于读锁也是如此。

上面写的有点啰嗦,我用大白话总结一下:我读数据时,你可以去读,因为我两的数据是一样的;我读数据时,你不能写,你写了,数据就变了,我还读个鬼啊;我写数据时,你不能读,也不能写,我就是这么强势。下面看一个实例:

var count intvar mutex sync.RWMutexfunc write(n int) {   rand.Seed(time.Now().UnixNano())   fmt.Printf("写 goroutine %d 正在写数据...\n", n)   mutex.Lock()   num := rand.Intn(500)   count = num   fmt.Printf("写 goroutine %d 写数据结束,写入新值 %d\n", n, num)   mutex.Unlock()}func read(n int) {   mutex.RLock()   fmt.Printf("读 goroutine %d 正在读取数据...\n", n)   num := count   fmt.Printf("读 goroutine %d 读取数据结束,读到 %d\n", n, num)   mutex.RUnlock()}func main() {   for i := 0; i < 10; i++ {      go read(i + 1)   }   for i := 0; i < 10; i++ {      go write(i + 1)   }   time.Sleep(time.Second*5)}//输出结果读 goroutine 1 正在读取数据...读 goroutine 1 读取数据结束,读到 0读 goroutine 7 正在读取数据...读 goroutine 7 读取数据结束,读到 0读 goroutine 3 正在读取数据...读 goroutine 3 读取数据结束,读到 0读 goroutine 10 正在读取数据...读 goroutine 10 读取数据结束,读到 0读 goroutine 8 正在读取数据...读 goroutine 8 读取数据结束,读到 0读 goroutine 6 正在读取数据...读 goroutine 5 正在读取数据...读 goroutine 5 读取数据结束,读到 0写 goroutine 2 正在写数据...读 goroutine 4 正在读取数据...读 goroutine 4 读取数据结束,读到 0写 goroutine 4 正在写数据...写 goroutine 3 正在写数据...读 goroutine 2 正在读取数据...读 goroutine 2 读取数据结束,读到 0写 goroutine 9 正在写数据...读 goroutine 6 读取数据结束,读到 0写 goroutine 7 正在写数据...读 goroutine 9 正在读取数据...读 goroutine 9 读取数据结束,读到 0写 goroutine 6 正在写数据...写 goroutine 1 正在写数据...写 goroutine 8 正在写数据...写 goroutine 10 正在写数据...写 goroutine 5 正在写数据...写 goroutine 2 写数据结束,写入新值 365写 goroutine 4 写数据结束,写入新值 47写 goroutine 3 写数据结束,写入新值 468写 goroutine 9 写数据结束,写入新值 155写 goroutine 7 写数据结束,写入新值 112写 goroutine 6 写数据结束,写入新值 490写 goroutine 1 写数据结束,写入新值 262写 goroutine 8 写数据结束,写入新值 325写 goroutine 10 写数据结束,写入新值 103写 goroutine 5 写数据结束,写入新值 353

可以看出前面10个协程可以并行读取数据,后面10个协程,就全部阻塞在了“...正在写数据...”过程,等读完了,然后10个协程就开始依次写。


9.3 总结


本章的内容不多,主要需要注意互斥锁和读写锁的几条注意事项,读写锁其实就是更细粒度的锁划分,为了能让程序更好并发,上面已经讲述的非常清楚,这里就不再啰嗦。唯一再强调的一点,无论是互斥锁还是读写锁,我们都不要试图去解锁未锁定的锁,因为这样会引发不可恢复的 panic。

相关文章
|
1天前
|
程序员 Go PHP
为什么大部分的 PHP 程序员转不了 Go 语言?
【9月更文挑战第8天】大部分 PHP 程序员难以转向 Go 语言,主要因为:一、编程习惯与思维方式差异,如语法风格和编程范式;二、学习成本高,需掌握新知识体系且面临项目压力;三、职业发展考量,现有技能价值及市场需求不确定性。学习新语言虽有挑战,但对拓宽职业道路至关重要。
19 10
|
1天前
|
算法 程序员 Go
PHP 程序员学会了 Go 语言就能唬住面试官吗?
【9月更文挑战第8天】学会Go语言可提升PHP程序员的面试印象,但不足以 solely “唬住” 面试官。学习新语言能展现学习能力、拓宽技术视野,并增加就业机会。然而,实际项目经验、深入理解语言特性和综合能力更为关键。全面展示这些方面才能真正提升面试成功率。
19 10
|
1天前
|
编译器 Go
go语言学习记录(关于一些奇怪的疑问)有别于其他编程语言
本文探讨了Go语言中的常量概念,特别是特殊常量iota的使用方法及其自动递增特性。同时,文中还提到了在声明常量时,后续常量可沿用前一个值的特点,以及在遍历map时可能遇到的非顺序打印问题。
|
2天前
|
存储 Shell Go
Go语言结构体和元组全面解析
Go语言结构体和元组全面解析
|
5天前
|
安全 大数据 Go
深入探索Go语言并发编程:Goroutines与Channels的实战应用
在当今高性能、高并发的应用需求下,Go语言以其独特的并发模型——Goroutines和Channels,成为了众多开发者眼中的璀璨明星。本文不仅阐述了Goroutines作为轻量级线程的优势,还深入剖析了Channels作为Goroutines间通信的桥梁,如何优雅地解决并发编程中的复杂问题。通过实战案例,我们将展示如何利用这些特性构建高效、可扩展的并发系统,同时探讨并发编程中常见的陷阱与最佳实践,为读者打开Go语言并发编程的广阔视野。
|
7天前
|
Go
golang语言之go常用命令
这篇文章列出了常用的Go语言命令,如`go run`、`go install`、`go build`、`go help`、`go get`、`go mod`、`go test`、`go tool`、`go vet`、`go fmt`、`go doc`、`go version`和`go env`,以及它们的基本用法和功能。
18 6
|
7天前
|
存储 Go
Golang语言基于go module方式管理包(package)
这篇文章详细介绍了Golang语言中基于go module方式管理包(package)的方法,包括Go Modules的发展历史、go module的介绍、常用命令和操作步骤,并通过代码示例展示了如何初始化项目、引入第三方包、组织代码结构以及运行测试。
15 3
|
9天前
|
缓存 安全 Java
如何利用Go语言提升微服务架构的性能
在当今的软件开发中,微服务架构逐渐成为主流选择,它通过将应用程序拆分为多个小服务来提升灵活性和可维护性。然而,如何确保这些微服务高效且稳定地运行是一个关键问题。Go语言,以其高效的并发处理能力和简洁的语法,成为解决这一问题的理想工具。本文将探讨如何通过Go语言优化微服务架构的性能,包括高效的并发编程、内存管理技巧以及如何利用Go生态系统中的工具来提升服务的响应速度和资源利用率。
|
9天前
|
Rust Linux Go
Rust/Go语言学习
Rust/Go语言学习
|
11天前
|
Java 数据库连接 数据库
携手前行:在Java世界中深入挖掘Hibernate与JPA的协同效应
【8月更文挑战第31天】Java持久化API(JPA)是一种Java规范,为数据库数据持久化提供对象关系映射(ORM)方法。JPA定义了实体类与数据库表的映射及数据查询和事务控制方式,确保不同实现间的兼容性。Hibernate是JPA规范的一种实现,提供了二级缓存、延迟加载等丰富特性,提升应用性能和可维护性。通过结合JPA和Hibernate,开发者能编写符合规范且具有高度可移植性的代码,并利用Hibernate的额外功能优化数据持久化操作。
25 0
下一篇
DDNS