Go lock重复加锁不会panic场景

文章展示了在Go1.20版本中遇到的死锁情况,通过一段测试代码解释了当尝试二次锁定已锁定的Mutex时,程序会引发fatalerror。然后,修改后的代码使用sync.WaitGroup避免了死锁,通过在goroutines中释放锁并在主goroutine中等待所有子goroutines完成来实现同步。
摘要由CSDN通过智能技术生成

Go 的版本是Go1.20

➜  ~ go version
go version go1.20.4 darwin/arm64

测试代码

package main

import (
	"sync"
)

func main() {
	var lock sync.Mutex
	defer lock.Unlock()
	lock.Lock() // 第一次锁定
	lock.Lock() // 第二次锁定
}

执行

➜ ~ go run main.go
fatal error: all goroutines are asleep - deadlock!

goroutine 1 [sync.Mutex.Lock]:
sync.runtime_SemacquireMutex(0x140000a0000?, 0x0?, 0x1400009cef8?)
	/usr/local/go/src/runtime/sema.go:77 +0x28
sync.(*Mutex).lockSlow(0x140000a0000)
	/usr/local/go/src/sync/mutex.go:171 +0x178
sync.(*Mutex).Lock(...)
	/usr/local/go/src/sync/mutex.go:90
main.main()
	/Users/tu/main.go:11 +0x108
exit status 2

修改测试代码

package main

import (
	"sync"
	"time"
)

func main() {
	var lock sync.Mutex
	group := sync.WaitGroup{}
	for i := 1; i < 10; i++ {
		group.Add(1)
		go func() {
			defer group.Done()
			time.Sleep(10 * time.Second)
		}()
	}
	defer lock.Unlock()
	lock.Lock() // 第一次锁定
	lock.Lock() // 第二次锁定
	group.Wait()
}

执行后阻塞了并没有panic

➜ ~ go run main.go
 

 

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值