Golang教程:goroutine协程
Posted 奔梦
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Golang教程:goroutine协程相关的知识,希望对你有一定的参考价值。
在上一篇中,我们讨论了并发,以及并发和并行的区别。在这篇教程中我们将讨论在Go中如何通过Go协程实现并发。
什么是协程
Go协程(Goroutine)是与其他函数或方法同时运行的函数或方法。可以认为Go协程是轻量级的线程。与创建线程相比,创建Go协程的成本很小。因此在Go中同时运行上千个协程是很常见的。
Go协程对比线程的优点
- 与线程相比,Go协程的开销非常小。Go协程的堆栈大小只有几kb,它可以根据应用程序的需要而增长和缩小,而线程必须指定堆栈的大小,并且堆栈的大小是固定的。
- Go协程被多路复用到较少的OS线程。在一个程序中数千个Go协程可能只运行在一个线程中。如果该线程中的任何一个Go协程阻塞(比如等待用户输入),那么Go会创建一个新的OS线程并将其余的Go协程移动到这个新的OS线程。所有这些操作都是 runtime 来完成的,而我们程序员不必关心这些复杂的细节,只需要利用 Go 提供的简洁的 API 来处理并发就可以了。
- Go 协程之间通过信道(channel)进行通信。信道可以防止多个协程访问共享内存时发生竟险(race condition)。信道可以想象成多个协程之间通信的管道。我们将在下一篇教程中介绍信道。
如何创建一个协程?
在函数或方法调用之前加上关键字 go
,这样便开启了一个并发的Go协程。
让我们创建一个协程:
package main import ( "fmt" ) func hello() { fmt.Println("Hello world goroutine") } func main() { go hello() fmt.Println("main function") }
第11行,go hello()
开启了一个新的协程。现在 hello()
函数将和 main()
函数一起运行。main
函数在单独的协程中运行,这个协程称为主协程。
运行这个程序,你将得到一个惊喜。
程序仅输出了一行文本: main function
。我们创建的协程发生了什么?我们需要了解Go协程的两个属性,以了解为什么发生这种情况。
- 当创建一个Go协程时,创建这个Go协程的语句立即返回。与函数不同,程序流程不会等待Go协程结束再继续执行。程序流程在开启Go协程后立即返回并开始执行下一行代码,忽略Go协程的任何返回值。
- 在主协程存在时才能运行其他协程,主协程终止则程序终止,其他协程也将终止。
我想你已经知道了为什么我们的协程为什么没有运行。在11行调用 go hello()
后,程序的流程直接调转到下一条语句执行,并没有等待 hello
协程退出,然后打印 main function
。接着主协程结束运行,不会再执行任何代码,因此 hello
协程没有得到运行的机会。
让我们修复这个问题:
package main import ( "fmt" "time" ) func hello() { fmt.Println("Hello world goroutine") } func main() { go hello() time.Sleep(1 * time.Second) fmt.Println("main function") }
上面的程序中,第13行,我们调用 time 包的 Sleep 函数来使调用该函数所在的协程休眠。在这里是让主协程休眠1秒钟。现在调用 go hello()
有了足够的时间得以在主协程退出之前执行。该程序首先打印 Hello world goroutine
,等待1秒钟之后打印 main function
。
在主协程中使用 Sleep 函数等待其他协程结束的方法是不正规的,我们用在这里只是为了说明Go协程是如何工作的。信道可以用于阻塞主协程,直到其他协程执行完毕。我们将在下一篇教程中讨论信道。
本文转自:https://blog.csdn.net/u011304970/article/details/75096323?locationNum=3&fps=1
以上是关于Golang教程:goroutine协程的主要内容,如果未能解决你的问题,请参考以下文章