go语言中的读写锁以及协程通信

0 需求分析

  • 在一个读操作远大于写操作的时候,用读写锁
  • 读写锁与互斥锁的运行时长相比
  • 两个goroutine间的通信,写goroutine写完以后通过channel发一个消息给读goroutine,读goroutine收到以后开启200个并发读
  • 两个goroutine间的通信用channel ,channel是阻塞的

1 源代码

package main

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

var (
	x         = 0
	wg        sync.WaitGroup
	myRwMutex sync.RWMutex
	myChannel chan int
)

//读操作
func myRead() {
	defer wg.Done()
	for i := 0; i < 100; i++ {
		//加读锁
		myRwMutex.RLock()
		//myRwMutex.Lock()
		fmt.Println(x)
		time.Sleep(time.Millisecond)
		myRwMutex.RUnlock()
		//myRwMutex.Unlock()
	}
}

//写操作
func myWrite(myChannel *chan int) {
	defer wg.Done()
	for i := 0; i < 200; i++ {
		//加写锁
		myRwMutex.Lock()
		x += 1
		time.Sleep(time.Millisecond * 5)
		myRwMutex.Unlock()
	}
	//写完成以后向channel发送标志位
	(*myChannel) <- 1
}

func main() {
	/*
		--在一个读操作远大于写操作的时候,用读写锁
		--读写锁与互斥锁的运行时长相比
		--两个goroutine间的通信,写goroutine写完以后通过channel发一个消息给
		  读goroutine,读goroutine收到以后开启200个并发读
	*/
	//用于两个goroutine间的通信
	myChannel = make(chan int, 1)
	nowTime := time.Now()

	//计数器为三,三个协程,一个写 两个读
	wg.Add(3)

	//写goroutine开始工作
	go myWrite(&myChannel)

	//判断myWrite是否完成
	flag, ok := <-myChannel
	if !ok || flag != 1 {
		fmt.Printf("myChannel中没有取到数据:%v\n", ok)
		return
	}

	//myWrite完成后开启读模式
	fmt.Printf("myChannel中取到数据了:%v\n", flag)
	close(myChannel)
	go myRead()
	go myRead()

	//等待计数器跑完后再终止main函数
	wg.Wait()

	//计算时间
	fmt.Println(time.Now().Sub(nowTime))

}

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

唐维康

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值