golang 协程 (goroutine) 与通道 (channel)

2024-03-08 01:20

本文主要是介绍golang 协程 (goroutine) 与通道 (channel),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

golang的协程和通道,之前就看过了,一直没有很好的理解,所以一直也没记录,今天看书,看到有一个总结的章节,里面记录了一些注意事项,因此写个文档,记录一下,避免以后自己忘了或者是找不见资料

顺便吐槽下公司的业务,自己负责的业务能啥也不知道,开发完了给他们上线了,完事还问你,这个为什么会这样,这不是你要求的吗?UAT的时候业务全程参加,都看过了,没问题才上线,过了一个月尽然能忘得一干二净。

出于性能考虑的建议:
实践经验表明,为了使并行运算获得高于串行运算的效率,在协程内部完成的工作量,必须远远高于协程的创建和相互来回通信的开销。

出于性能考虑建议使用带缓存的通道:
使用带缓存的通道可以很轻易成倍提高它的吞吐量,某些场景其性能可以提高至 10 倍甚至更多。通过调整通道的容量,甚至可以尝试着更进一步的优化其性能。

限制一个通道的数据数量并将它们封装成一个数组:
如果使用通道传递大量单独的数据,那么通道将变成性能瓶颈。然而,将数据块打包封装成数组,在接收端解压数据时,性能可以提高至 10 倍。

现在创建一个带缓存的通道:ch := make(chan type,buf)
(1)如何使用 for 或者 for-range 遍历一个通道:(尽量使用这种或者是跟select配合使用)

这种其实就是一个for循环遍历通道,但是golang的机制,这里会自动监测通道是否关闭,而不需要开发二次判断通道是否关闭
但是这里有个坑需要注意,会有死锁的问题,因为你的通道中没有数据的时候,for range ch 会发生阻塞,但是无法解除阻塞,发生死锁

for v := range ch {// do something with v
}

(2)如何检测一个通道 ch 是否关闭:

//read channel until it closes or error-condition
for {if input, open := <-ch; !open {// 这里!open,就是表示通道已经被关了,break跳出循环,不从通道里面获取数据了break}fmt.Printf("%s", input)
}

(3)如何通过一个通道让主程序等待直到协程完成(信号量模式):如果希望程序一直阻塞,在匿名函数中省略 ch <- 1 即可。

ch := make(chan int) // Allocate a channel.
// Start something in a goroutine; when it completes, signal on the channel.
go func() {// doSomethingch <- 1 // Send a signal; value does not matter.
}()
doSomethingElseForAWhile()
<-ch // Wait for goroutine to finish; discard sent value.
func compute(ch chan int){ch <- someComputation() // when it completes, signal on the channel.
}func main(){ch := make(chan int) 	// allocate a channel.go compute(ch)		// start something in a goroutinesdoSomethingElseForAWhile()result := <- ch
}

(4)通道的工厂模板:以下函数是一个通道工厂,启动一个匿名函数作为协程以生产通道:

func pump() chan int {ch := make(chan int)go func() {for i := 0; ; i++ {ch <- i}}()return ch
}

(5)通道迭代器模板:

func (c *container) Iter () <- chan item {ch := make(chan item)go func () {for i:= 0; i < c.Len(); i++{	// or use a for-range loopch <- c.items[i]}} ()return ch
}
for x := range container.Iter() { ... }

(6)如何限制并发处理请求的数量

package mainconst MAXREQS = 50var sem = make(chan int, MAXREQS)type Request struct {a, b   intreplyc chan int
}func process(r *Request) {// do something
}func handle(r *Request) {sem <- 1 // doesn't matter what we put in itprocess(r)<-sem // one empty place in the buffer: the next request can start
}func server(service chan *Request) {for {request := <-servicego handle(request)}
}func main() {service := make(chan *Request)go server(service)
}

(7)如何在多核CPU上实现并行计算:

func DoAll(){sem := make(chan int, NCPU) // Buffering optional but sensiblefor i := 0; i < NCPU; i++ {go DoPart(sem)}// Drain the channel sem, waiting for NCPU tasks to completefor i := 0; i < NCPU; i++ {<-sem // wait for one task to complete}// All done.
}func DoPart(sem chan int) {// do the part of the computationsem <-1 // signal that this piece is done
}func main() {runtime.GOMAXPROCS(NCPU) // runtime.GOMAXPROCS = NCPUDoAll()
}

(8)如何终止一个协程:runtime.Goexit()

(9)简单的超时模板:

timeout := make(chan bool, 1)
go func() {time.Sleep(1e9) // one second  timeout <- true
}()
select {case <-ch:// a read from ch has occurredcase <-timeout:// the read from ch has timed out
}

(10)如何使用输入通道和输出通道代替锁:

func Worker(in, out chan *Task) {for {t := <-inprocess(t)out <- t}
}

(11)如何在同步调用运行时间过长时将之丢弃:

// 注意缓冲大小设置为 1 是必要的,可以避免协程死锁以及确保超时的通道可以被垃圾回收。
// 此外,需要注意在有多个 case 符合条件时, select 对 case 的选择是伪随机的
// 如果代码稍作修改如下
// 则 select 语句可能不会在定时器超时信号到来时立刻选中 time.After(timeoutNs) 对应的 case
// 因此协程可能不会严格按照定时器设置的时间结束。
ch := make(chan int, 1)
go func() { for { ch <- 1 } } ()
L:
for {select {case <-ch:// do somethingcase <-time.After(timeoutNs):// call timed outbreak L}
}

(12)如何在通道中使用计时器和定时器:定时器 (Timer) 结构体和计时器 (Ticker) 结构体

package mainimport ("fmt""time"
)func main() {tick := time.Tick(1e8)boom := time.After(5e8)for {select {case <-tick:fmt.Println("tick.")case <-boom:fmt.Println("BOOM!")returndefault:fmt.Println("    .")time.Sleep(5e7)}}
}

这篇关于golang 协程 (goroutine) 与通道 (channel)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/785493

相关文章

golang中reflect包的常用方法

《golang中reflect包的常用方法》Go反射reflect包提供类型和值方法,用于获取类型信息、访问字段、调用方法等,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值... 目录reflect包方法总结类型 (Type) 方法值 (Value) 方法reflect包方法总结

Golang如何对cron进行二次封装实现指定时间执行定时任务

《Golang如何对cron进行二次封装实现指定时间执行定时任务》:本文主要介绍Golang如何对cron进行二次封装实现指定时间执行定时任务问题,具有很好的参考价值,希望对大家有所帮助,如有错误... 目录背景cron库下载代码示例【1】结构体定义【2】定时任务开启【3】使用示例【4】控制台输出总结背景

Golang如何用gorm实现分页的功能

《Golang如何用gorm实现分页的功能》:本文主要介绍Golang如何用gorm实现分页的功能方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录背景go库下载初始化数据【1】建表【2】插入数据【3】查看数据4、代码示例【1】gorm结构体定义【2】分页结构体

在Golang中实现定时任务的几种高效方法

《在Golang中实现定时任务的几种高效方法》本文将详细介绍在Golang中实现定时任务的几种高效方法,包括time包中的Ticker和Timer、第三方库cron的使用,以及基于channel和go... 目录背景介绍目的和范围预期读者文档结构概述术语表核心概念与联系故事引入核心概念解释核心概念之间的关系

Golang 日志处理和正则处理的操作方法

《Golang日志处理和正则处理的操作方法》:本文主要介绍Golang日志处理和正则处理的操作方法,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考... 目录1、logx日志处理1.1、logx简介1.2、日志初始化与配置1.3、常用方法1.4、配合defer

golang float和科学计数法转字符串的实现方式

《golangfloat和科学计数法转字符串的实现方式》:本文主要介绍golangfloat和科学计数法转字符串的实现方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望... 目录golang float和科学计数法转字符串需要对float转字符串做处理总结golang float

golang实现延迟队列(delay queue)的两种实现

《golang实现延迟队列(delayqueue)的两种实现》本文主要介绍了golang实现延迟队列(delayqueue)的两种实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的... 目录1 延迟队列:邮件提醒、订单自动取消2 实现2.1 simplChina编程e简单版:go自带的time

Golang实现Redis分布式锁(Lua脚本+可重入+自动续期)

《Golang实现Redis分布式锁(Lua脚本+可重入+自动续期)》本文主要介绍了Golang分布式锁实现,采用Redis+Lua脚本确保原子性,持可重入和自动续期,用于防止超卖及重复下单,具有一定... 目录1 概念应用场景分布式锁必备特性2 思路分析宕机与过期防止误删keyLua保证原子性可重入锁自动

golang 对象池sync.Pool的实现

《golang对象池sync.Pool的实现》:本文主要介绍golang对象池sync.Pool的实现,用于缓存和复用临时对象,以减少内存分配和垃圾回收的压力,下面就来介绍一下,感兴趣的可以了解... 目录sync.Pool的用法原理sync.Pool 的使用示例sync.Pool 的使用场景注意sync.

golang中slice扩容的具体实现

《golang中slice扩容的具体实现》Go语言中的切片扩容机制是Go运行时的一个关键部分,它确保切片在动态增加元素时能够高效地管理内存,本文主要介绍了golang中slice扩容的具体实现,感兴趣... 目录1. 切片扩容的触发append 函数的实现2. runtime.growslice 函数gro