go语言中实现协程功能的两种方式及示例代码

发布时间:2022-6-10 10:42

协程本质上是一种用户态线程,不需要操作系统来进行抢占式调度,并且在真正的实现中寄存于线程中,因此系统开销极小,可以有效的提高线程任务的并发性,而避免多线程的缺点。

协程的优点:使用协程的优点是编程简单,结构清晰;

协程的缺点:需要语言的支持,如果不支持,则需要用户在程序中自行实现调度器。

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:

通过该方式,我们能够确保协程进行执行。

但强烈推荐不要使用该方式。

存在协程执行完,主线程的睡眠时间还未到,程序还是处于执行状态。

同时,如果主线程睡眠时间到了,但是协程任务还未执行完毕,剩下的任务就不会被执行。

 

Go语言入门之函数的定义与使用 生活杂谈

Go语言入门之函数的定义与使用

函数是一段代码的片段,包含连续的执行语句,它可以将零个或多个输入参数映射到零个或多个参数输出。函数像一个黑盒,对它的使用者隐藏实现细节。还可以在代码中通过函数调用来执行它们。 学到现在,我们使用...
利用Go语言快速实现一个极简任务调度系统 网站建设

利用Go语言快速实现一个极简任务调度系统

任务调度(Task Scheduling)是很多软件系统中的重要组成部分,字面上的意思是按照一定要求分配运行一些通常时间较长的脚本或程序。在爬虫管理平台 Crawlab 中,任务调度是其中的核心模块,...
Go语言select语句用法示例 网站建设

Go语言select语句用法示例

多个通道 Channel 中信息的发送和接受处理的专用的语句—select 语句。select 语句会阻塞,直到其中的一个发送/接收操作准备好。select 语句和 switch 语句有点相似,但 s...
go mock server的简易实现示例 生活杂谈

go mock server的简易实现示例

学习golang也一段时间了,看了一些书,上周又看了一本入门级的《Go语言趣学指南》,是时候检验成果了。 目的:通过读取本地mock数据,发起http请求,返回给前端,实现mock功能。 代码...