有志者,事竟成!如果你在学习Golang,那么本文《需要一点帮助来理解代码流程吗?我不明白输出中的例程结束如何出现在其他输出语句之间》,就很适合你!文章讲解的知识点主要包括,若是你对本文感兴趣,或者是想搞懂其中某个知识点,就请你继续往下看吧~
问题内容试图理解 goroutine 的流程,所以我写了这段代码,只有一件事我无法理解,那就是,routine-end 如何在其他 go 例程之间运行,并完成单个 go 例程并打印通道的输出在最后。
import(
"fmt"
)
func add(dataarr []int,datachannel chan int,i int ){
var sum int
fmt.println("goroutine",i+1)
for i:=0;i<len(dataarr);i++{
sum += dataarr[i]
}
fmt.println("writing to channel.....")
datachannel <- sum
fmt.println("routine-end")
}
func main(){
fmt.println("main() started")
datachannel := make(chan int)
dataarr := []int{1,2,3,4,5,6,7,8,9}
for i:=0;i<len(dataarr);i+=3{
go add(dataarr[i:i+3],datachannel,i)
}
fmt.println("came to blocking statement ..........")
fmt.println(<-datachannel)
fmt.println("main() end")
}
output
main() started
came to blocking statement ..........
GOROUTINE 1
wRITING TO CHANNEL.....
routine-end
GOROUTINE 4
wRITING TO CHANNEL.....
6
main() end
解决方案
您的 for
循环启动 3 个调用 add
函数的 goroutine。
此外,main
本身运行在一个单独的“主”goroutine 中。
由于 goroutine 是并发执行的,因此它们的运行顺序通常是不可预测的,并且取决于时间、机器的繁忙程度等。运行之间和机器之间的结果可能会有所不同。在不同位置插入 time.Sleep
调用可能有助于可视化它。例如,在“来到阻塞语句”之前插入 time.Sleep
100ms,显示所有 add
Goroutines 启动。
您在运行中通常会看到的是,一个 add
Goroutine 启动,将其切片添加到其 sum
并将 sum
写入 dataChannel
。由于 main
启动了一些 goroutines 并立即从通道中读取,因此这次读取获取了 add
写入的 add
,然后程序存在 - 因为默认情况下 main 不会等待所有 goroutine 完成。
此外,由于 dataChannel
通道是无缓冲的,并且 main
只读取一个值,因此其他 add
goroutine 在写入时将无限期地阻塞在该通道上。
我确实建议您查看一些有关 goroutine 和通道的介绍性资源。他们从简单的原则出发建立概念。一些适合您的好链接:
- Golang tour
- https://gobyexample.com/ -- 从 Goroutines 示例开始,然后执行接下来的几个示例。
今天关于《需要一点帮助来理解代码流程吗?我不明白输出中的例程结束如何出现在其他输出语句之间》的内容介绍就到此结束,如果有什么疑问或者建议,可以在编程网公众号下多多回复交流;文中若有不正之处,也希望回复留言以告知!