做网站排名费用多少钱,免费做网页的网站,seo关键词排名优化软件怎么选,python下载安装目录
关闭channel
引入 不关闭通道是有风险的#xff0c;主要存在两条#xff1a;
如何优雅的关闭#xff1f;
1.只有一个发送者
2.多个发送者
判断通道是否关闭
有缓存和无缓存的通道有什么区别#xff1f;
错误的关闭通道 关闭channel close(chan)#xff1a;关…目录
关闭channel
引入 不关闭通道是有风险的主要存在两条
如何优雅的关闭
1.只有一个发送者
2.多个发送者
判断通道是否关闭
有缓存和无缓存的通道有什么区别
错误的关闭通道 关闭channel close(chan)关闭通道并不会丢失里面的数据只是让读取通道数据的时候不会读完之后一直阻塞等待新数据写入。 引入
看一看下面的例子
ch : make(chan int, 1)
ch - 11
// close(ch) // 如果不关闭下面会引发死锁// 通道没有关闭的时候会阻塞程序会永远等待从通道中接收值。而且由于没有其他goroutine来关闭通道这个阻塞状态将会持续下去从而引发死锁。
for v : range ch {fmt.Println(v)
} 不关闭通道是有风险的主要存在两条
出现死锁如果通道不会再被使用但未关闭接收者可能会一直等待数据导致死锁或无意义的等待。出现资源泄漏未关闭的通道可能会导致资源泄漏因为垃圾收集器不会回收未关闭的通道。而且还要消耗电脑的性能来维护这条管道。 如何优雅的关闭
在关闭管道的时候我们要注意正确地决定何时关闭通道、确保通道不会被多次关闭。多次关闭同一个通道会报panic 1.只有一个发送者
如果只有一个发送者那么这个发送者在完成所有发送操作后可以直接关闭通道
package mainimport (fmttime
)// 边入边出
func main() {c : make(chan int, 5)go fibonacci(10, c)for v : range c {fmt.Println(out:, time.Now())fmt.Println(v)}
}// 应该由发送数据的一方关闭通道当数据发送完毕后就是关闭通道的时候。
func fibonacci(n int, c chan int) {x, y : 0, 1for i : 0; i n; i {c - xfmt.Println(in:, time.Now())time.Sleep(100)x, y y, xy}close(c)
}// 大致运行结果
// in: 2024-07-16 16:00:34.637655 0800 CST m0.000171297
// out: 2024-07-16 16:00:34.637681 0800 CST m0.000197020
// in: 2024-07-16 16:00:34.637952 0800 CST m0.000467725
// 0
// in: 2024-07-16 16:00:34.637963 0800 CST m0.000478508
// out: 2024-07-16 16:00:34.637968 0800 CST m0.000483915
// 1
// in: 2024-07-16 16:00:34.637976 0800 CST m0.000491801
// in: 2024-07-16 16:00:34.637983 0800 CST m0.000499051
// out: 2024-07-16 16:00:34.637978 0800 CST m0.000494157
// 1
// out: 2024-07-16 16:00:34.637999 0800 CST m0.000514485
// 2
// out: 2024-07-16 16:00:34.638006 0800 CST m0.000522241
// 3
// out: 2024-07-16 16:00:34.638013 0800 CST m0.000528591
// 5
// in: 2024-07-16 16:00:34.637988 0800 CST m0.000504017
// in: 2024-07-16 16:00:34.638039 0800 CST m0.000555021
// in: 2024-07-16 16:00:34.638045 0800 CST m0.000561340
// in: 2024-07-16 16:00:34.638049 0800 CST m0.000564725
// in: 2024-07-16 16:00:34.638055 0800 CST m0.000570642
// out: 2024-07-16 16:00:34.638052 0800 CST m0.000568007
// 8
// out: 2024-07-16 16:00:34.638095 0800 CST m0.000610606
// 13
// out: 2024-07-16 16:00:34.638103 0800 CST m0.000619251
// 21
// out: 2024-07-16 16:00:34.63811 0800 CST m0.000626188
// 34 2.多个发送者
当有多个发送者时可以使用 sync.WaitGroup 来协调这些发送者并在所有发送者完成后由一个协程关闭通道
package mainimport (fmtsync
)func main() {ch : make(chan string)var wg sync.WaitGroup// 启动3个协程发送数据for i : 1; i 5; i {wg.Add(1)go func(num int) {defer wg.Done()for j : a; j e; j {ch - fmt.Sprintf(协程%d:%c, num, j)}}(i)}// 启动一个协程来等待所有发送者完成并关闭通道go func() {wg.Wait()close(ch)}()// 接收数据for val : range ch {fmt.Println(val)}fmt.Println(通道已关闭结束接收)
}判断通道是否关闭 v, ok : -ch 如果 ok 为 true表示成功从 channel 中接收到一个值并且 channel 还没有关闭。如果 ok 为 false表示 channel 已经关闭。
当一个 channel 已经关闭而且其中的元素已经全部被取出时再从管道中取出数据会返回该元素类型的零值并且 ok 会被设置为 false。这样的检查是为了防止在已关闭的 channel 上进行接收操作时引发 panic。因为在已关闭的 channel 上进行接收操作会立即返回零值但如果不进行检查可能会误认为是从 channel 中接收到了有效的数据。因此通过检查 ok我们可以确定是否成功接收到了有效的值。 c : make(chan int, 5)c - 1c - 2c - 3close(c)v1, ok : -cfmt.Println(v1, ok) // 1 truev1, ok -cfmt.Println(v1, ok) // 2 truev1, ok -cfmt.Println(v1, ok) // 3 truev1, ok -cfmt.Println(v1, ok) // 0 falsev1, ok -cfmt.Println(v1, ok) // 0 false 有缓存和无缓存的通道有什么区别
参考下面链接
golang channel有无缓冲区的区别 错误的关闭通道 错误一多次关闭同一个通道 c : make(chan int, 5)for i : 0; i 5; i {defer close(c) // 报panic同一个通道关闭了多次}
错误二资源泄漏 for i : 0; i 5; i {c : make(chan int, 5)defer close(c)}
看起来没有错误运行出来也没有错误。但是可能会导致资源泄漏因为每次循环迭代中创建的通道不会立即关闭导致内存和其他资源的泄漏。这五个管道的关闭时间都是在所在函数运行完毕的时候关闭的。而不是本次循环结束后就立刻关闭。