周末又到了,为你们准备了一份实用干货:如何使用channel和Mutex解决并发问题,利用周末的好时光,配上音乐,思考一下吧🤔。html
来,问本身个问题:面对并发问题,是用channel解决,仍是用Mutex解决?git
若是本身内心尚未清晰的答案,那就读下这篇文章,你会了解到:github
前面不少篇的文章都在围绕channel
介绍,而只有前一篇sync
的文章介绍到了Mutex
,不是我偏爱,而是channel在Golang是first class级别的,设计在语言特性中的,而Mutex
只是一个包中的。这就注定了一个是主角,一个是配角。golang
而且Golang还有一个并发座右铭,在《Effective Go》的channel介绍中写到:缓存
Share memory by communicating, don't communicate by sharing memory.
经过通讯共享内存,而不是经过共享内存而通讯。
Golang以如此明显的方式告诉咱们:面对并发问题,你首先想到的应该是channel,由于channel是线程安全的而且不会有数据冲突,比锁好用多了。安全
既生瑜,何生亮。既然有channel
了,为啥还提供sync.Mutex
呢?并发
主角不是万能的,他也须要配角。在Golang里,channel也不是万能的,这是由channel的特性和局限形成的。less
下面就给你们介绍channel的特色、核心方法和缺点。异步
channel的核心是数据流动,关注到并发问题中的数据流动,把流动的数据放到channel中,就能使用channel解决这个并发问题。这个思路是从Go语言的核心开发者的演讲中学来的,然而视频我已经找不到了,否则直接共享给你们,他提到了Golang并发的核心实践的4个点:函数
DataFlow -> Drawing -> Pipieline -> Exiting
DataFlow指数据流动,Drawing指把数据流动画出来,Pipeline指的是流水线,Exit指协程的退出。DataFlow + Drawing就是我提到到channel解决并发问题的思路,Pipeline和Exit是具体的实践模式,Pipeline和Exit我都写过文章,有须要自取:
下面我使用例子具体解释DataFlow + Drawing。借用《Golang并发的次优选择:sync包》中银行的例子,介绍如何使用channel解决例子中银行的并发问题:银行支持多个用户的同时操做。顺便看下同一个并发问题,使用channel和Mutex解决是什么差异。
一块儿分析下多个用户同时操做银行的数据流动:
你必定发现了上面的数据流动:
channel是数据流动的通道/管道,为流动的数据创建通道,这里须要创建2类channel:
咱们把channel添加到上图中,获得下面的图:
以上就是从数据流动的角度,发现如何使用channel解决并发问题。思路有了,再思考下代码层面须要怎么作:
银行:
以上,咱们这个并发问题的逻辑实现和各块工做就清晰了,写起来也方便、简单。代码实现有200多行,公众号不方便查看,能够点阅读原文,一键直达。
代码不能贴了,运行结果仍是能够的,为了方便理解结果,介绍下示例代码作了什么。main函数建立了银行、小明、小刚3个并发协程:
reqCh
接收请求,依次处理每一个请求,直到通道关闭,把请求交给处理函数,处理函数把结果写入到请求中的retCh
。reqCh
。main函数最后使用WaitGroup等待小明、小刚结束后退出。
下面是运行结果:
$ go run channel_map.go xiaogang deposite 100 success xiaoming deposite 100 success xiaogang withdraw 200 failed xiaoming withdraw 20 success xiaogang has 100 xiaoming has 80 Bank exit
这一遭搞完,发现啥没有?用Mutex直接加锁、解锁完事了,但channel搞出来一坨,是否是用channel解决这个问题不太适合?是的。对于当前这个问题,和Mutex的方案相比,channel的方案显的有点“重”,不够简洁、高效、易用。
但这个例子展现了3点:
如今,回到了开篇的问题:同一个并发问题,你是用channel解决,仍是用mutex解决?下面,一块儿看看怎么选择。
面对一个并发问题的时候,应当选择合适的并发方式:channel仍是mutex。选择的依据是他们的能力/特性:channel的能力是让数据流动起来,擅长的是数据流动的场景,《Channel or Mutex》中给了3个数据流动的场景:
mutex的能力是数据不动,某段时间只给一个协程访问数据的权限擅长数据位置固定的场景,《Channel or Mutex》中给了2个数据不动场景:
map
就是一种状态提供解决并发问题的一个思路:
面对并发问题,除了channel or mutex,你还有另一个选择:channel plus mutex。
一个大并发问题,能够分解成不少小的并发问题,每一个小的并发均可以单独选型:channel or mutex。但对于整个大的问题,一般不是channel or mutex,而是channel plus mutex。
若是你是认为是channel and mutex也行,但我更喜欢plus,体现相互配合。
读到这里,感受这篇文章头重脚轻,channel的讲了不少,而channel和mutex的选择却讲的不多。在channel和mutex的选择,实际并无一个固定答案,也没有固定的方法,但提供了一个简单的思路:设计出channel和Mutex的简单方案,而后选择最适合当前业务、问题的那个。
思考比结论更重要,但愿你有所收获:
- 若是这篇文章对你有帮助,请点个赞/喜欢,感谢。
- 本文做者:大彬
- 若是喜欢本文,随意转载,但请保留此原文连接:http://lessisbetter.site/2019/01/14/golang-channel-and-mutex/