go语言中实现协程功能的两种方式及示例代码
协程介绍
协程本质上是一种用户态线程,不需要操作系统来进行抢占式调度,
并且在真正的实现中寄存于线程中,因此系统开销极小,可以有效的提高线程任务的并发性,
而避免多线程的缺点。
协程的优点:
使用协程的优点是编程简单,结构清晰;
协程的缺点:
缺点是需要语言的支持,如果不支持,则需要用户在程序中自行实现调度器。
go语言中实现协程是非常简单的
实现协程方式一 :
我们可以通过线程之前同步的方式。
也就是说每个协程执行结束之后,主线程才结束。
package main
import (
"fmt"
"sync"
)
var wg sync.WaitGroup
func Hello(i int) {
defer wg.Done() // 减少WaitGroup计数器的值,应在线程的最后执行。
fmt.Println("Hello Goroutine!", i)
}
func main() {
for i := 0; i < 10; i++ {
wg.Add(1) // 向协程计数器加1。
go Hello(i)
}
wg.Wait() // 阻塞协程,直到协程计数器为0。
fmt.Println("所有的协程已经结束了,主线程可以结束了。")
}
go run goroutine.go
Hello Goroutine! 2
Hello Goroutine! 9
Hello Goroutine! 0
Hello Goroutine! 3
Hello Goroutine! 7
Hello Goroutine! 1
Hello Goroutine! 8
Hello Goroutine! 5
Hello Goroutine! 4
Hello Goroutine! 6
所有的协程已经结束了,主线程可以结束了。
ps:
通过上面的输出结果,我们可以看到协程对应的内容进行输出了。
同时,我们可以看到启动的协程并未是按照顺序输出的,而是随机输出。
是因为协程在执行过程中就是随机执行,而不是顺序执行。
实现协程方式二 :
该方式是通过睡眠时间,就是让主线程处于睡眠等待而不是立即结束,
在等待的时间段内,协程就可以执行。
package main
import (
"fmt"
"time"
)
func Show(ch1 chan int) {
val := <-ch1
fmt.Println("通道的值是:", val)
}
func main() {
ch1 := make(chan int, 10)
for i := 0; i < 10; i++ {
go Show(ch1)
ch1 <- i
}
time.Sleep(time.Second * 10)
close(ch1)
fmt.Println("所有的协程已经结束了,主线程可以结束了。")
}
go run goroutine.go
通道的值是: 4
通道的值是: 0
通道的值是: 1
通道的值是: 7
通道的值是: 9
通道的值是: 6
通道的值是: 8
通道的值是: 3
通道的值是: 2
通道的值是: 5
所有的协程已经结束了,主线程可以结束了。
ps:
通过该方式,我们能够确保协程进行执行。
但强烈推荐不要使用该方式。
存在协程执行完,主线程的睡眠时间还未到,程序还是处于执行状态。
同时,如果主线程睡眠时间到了,但是协程任务还未执行完毕,剩下的任务就不会被执行。
相关文章