Golang-goroutine互斥锁与读写互斥锁

package main

// go build -race main.go 编译后运行查看

import (
	"fmt"
	"sync"
	"time"
)

var wg sync.WaitGroup
var mutex sync.Mutex

var m = make(map[int]int, 0)

func test(num int) {
	mutex.Lock()
	var sum = 1
	for i := 1; i <= num; i++ {
		sum *= i
	}
	m[num] = sum
	fmt.Printf("key=%v value=%v\n", num, sum)
	time.Sleep(time.Millisecond)
	mutex.Unlock()
	wg.Done()

}

func main() {
	for r := 0; r < 40; r++ {
		wg.Add(1)
		go test(r)
	}
	wg.Wait()
}
package main

// go build -race main.go 编译后运行查看

import (
	"fmt"
	"sync"
	"time"
)

var wg sync.WaitGroup
var mutex sync.RWMutex

func read() {
	mutex.RLock()
	fmt.Println("---执行读操作---")
	time.Sleep(time.Second * 2)
	mutex.RUnlock()
	wg.Done()
}

func write() {
	mutex.Lock()
	fmt.Println("执行写操作")
	time.Sleep(time.Second * 2)
	mutex.Unlock()
	wg.Done()

}

func main() {
	// 开启10个协程执行读操作
	for i := 0; i < 40; i++ {
		wg.Add(1)
		go read()
	}
	// 开启10个协程执行写操作
	for i := 0; i < 10; i++ {
		wg.Add(1)
		go write()
	}
	wg.Wait()
}

Golang互斥锁读写锁存在多方面区别: - **并发控制机制**:互斥锁是完全互斥的,使用互斥锁能够保证同一时间有且只有一个 goroutine 进入临界区,其他的 goroutine 则在等待锁;当互斥锁释放后,等待的 goroutine 才可以获取锁进入临界区,多个 goroutine 同时等待一个锁时,唤醒的策略是随机的。而读写锁允许多个协程同时读取某个共享资源,但在写入时必须互斥,只能有一个协程进行写操作 [^1][^3]。 - **适用场景**:互斥锁适用于对共享资源的读写操作都需要严格互斥的场景。读写锁则更适用于读多写少的场景,当并发读取一个资源且不涉及资源修改时无需加锁,这种场景使用读写锁是更好的选择 [^1][^3]。 - **性能表现**:读写互斥锁在高并发读的场景下可以提高并发性能,但在高并发写的场景下仍然存在性能瓶颈。有人怀疑 golang 中 `sync.Mutex` 的 `Lock` 和 `Unlock` 在底层实现的时候要比 `sync.RWMutex` 的 `RLock` 和 `RUnlock` 性能要好 [^1][^2]。 以下是使用互斥锁读写锁的示例代码: ```go package main import ( "fmt" "math/rand" "sync" "time" ) var count int var mutex sync.RWMutex func main() { fmt.Println("start study golang...... ") 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) } func write(n int) { rand.Seed(time.Now().UnixNano()) fmt.Errorf("写 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() } ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值