Golang Goroutine的使用
作者:李培冠 发布时间:2023-09-20 20:38:45
什么是 Goroutine
goroutine 是 Go 并行设计的核心。goroutine 说到底其实就是协程,它比线程更小,十几个 goroutine 可能体现在底层就是五六个线程,Go 语言内部帮你实现了这些 goroutine 之间的内存共享。
执行 goroutine 只需极少的栈内存(大概是4~5KB),当然会根据相应的数据伸缩。也正因为如此,可同时运行成千上万个并发任务。goroutine 比 thread 更易用、更高效、更轻便。
一般情况下,一个普通计算机跑几十个线程就有点负载过大了,但是同样的机器却可以轻松地让成百上千个 goroutine 进行资源竞争。
Goroutine 的创建
只需在函数调⽤语句前添加 go 关键字,就可创建并发执⾏单元。
开发⼈员无需了解任何执⾏细节,调度器会自动将其安排到合适的系统线程上执行。
在并发编程中,我们通常想将一个过程切分成几块,然后让每个 goroutine 各自负责一块工作,当一个程序启动时,主函数在一个单独的 goroutine 中运行,我们叫它 main goroutine。新的 goroutine 会用 go 语句来创建。而 go 语言的并发设计,让我们很轻松就可以达成这一目的。
例如:
package main
import (
"fmt"
"time"
)
func foo() {
i := 0
for true {
i++
fmt.Println("new goroutine: i = ", i)
time.Sleep(time.Second)
}
}
func main() {
// 创建一个 goroutine, 启动另外一个任务
go foo()
i := 0
for true {
i++
fmt.Println("main goroutine: i = ", i)
time.Sleep(time.Second)
}
}
结果:
main goroutine: i = 1
new goroutine: i = 1
new goroutine: i = 2
main goroutine: i = 2
main goroutine: i = 3
new goroutine: i = 3
...
Goroutine 特性
主go程 退出后,其它的 子go程 也会自动退出:
package main
import (
"fmt"
"time"
)
func foo() {
i := 0
for true {
i++
fmt.Println("new goroutine: i = ", i)
time.Sleep(time.Second)
}
}
func main() {
// 创建一个 goroutine, 启动另外一个任务
go foo()
time.Sleep(time.Second * 3)
fmt.Println("main goroutine exit")
}
运行结果:
new goroutine: i = 1
new goroutine: i = 2
new goroutine: i = 3
main goroutine exit
runtime 包
Gosched
runtime.Gosched() 用于出让当前 go程 所占用的 CPU 时间片,让出当前 goroutine 的执行权限,调度器安排其他等待的任务运行,并在下次再获得 cpu 时间轮片的时候,从该出让 cpu 的位置恢复执行。
有点像跑接力赛,A 跑了一会碰到代码 runtime.Gosched() 就把接力棒交给 B 了,A 歇着了,B 继续跑。
例如:
package main
import (
"fmt"
"runtime"
"time"
)
func main() {
// 创建一个 goroutine
go func(s string) {
for i := 0; i < 2; i++ {
fmt.Println(s)
}
}("world")
for i := 0; i < 2; i++ {
runtime.Gosched()
fmt.Println("hello")
}
time.Sleep(time.Second * 3)
}
运行结果:
world
world
hello
hello
如果没有 runtime.Gosched() 则运行结果如下:
hello
hello
world
world
注意: runtime.Gosched() 只是出让一次机会,看下面的代码,注意运行结果:
package main
import (
"fmt"
"runtime"
"time"
)
func main() {
// 创建一个 goroutine
go func(s string) {
for i := 0; i < 2; i++ {
fmt.Println(s)
time.Sleep(time.Second)
}
}("world")
for i := 0; i < 2; i++ {
runtime.Gosched()
fmt.Println("hello")
}
}
运行结果:
world
hello
hello
为什么 world 只有一次呢?因为之前我们说过,主 goroutine 退出后,其它的工作 goroutine 也会自动退出。
Goexit
调用 runtime.Goexit() 将立即终止当前 goroutine 执⾏,调度器确保所有已注册 defer 延迟调用被执行。
注意与 return 的区别,return 是返回当前函数调用给调用者。
例如:
package main
import (
"fmt"
"runtime"
"time"
)
func main() {
go func() {
defer fmt.Println("A.defer")
func() {
defer fmt.Println("B.defer")
runtime.Goexit() // 终止当前 goroutine
fmt.Println("B") // 不会执行
}()
fmt.Println("A") // 不会执行
}() // 不要忘记 ()
time.Sleep(time.Second * 3)
}
运行结果:
B.defer
A.defer
GOMAXPROCS
调用 runtime.GOMAXPROCS() 用来设置可以并行计算的 CPU 核数的最大值,并返回 上一次(没有则是电脑默认的) 设置的值。
package main
import (
"fmt"
"runtime"
)
func main() {
runtime.GOMAXPROCS(1) // 将 cpu 设置为单核
for true {
go fmt.Print(0) // 子 go 程
fmt.Print(1) // 主 go 程
}
}
运行结果:
111111 ... 1000000 ... 0111 ...
在执行 runtime.GOMAXPROCS(1) 时,最多同时只能有一个 goroutine 被执行。所以会打印很多 1。过了一段时间后,GO 调度器会将其置为休眠,并唤醒另一个 goroutine,这时候就开始打印很多 0 了,在打印的时候,goroutine 是被调度到操作系统线程上的。
package main
import (
"fmt"
"runtime"
)
func main() {
runtime.GOMAXPROCS(2)
for true {
go fmt.Print(0)
fmt.Print(1)
}
}
运行结果:
111111111111111000000000000000111111111111111110000000000000000011111111100000...
在执行 runtime.GOMAXPROCS(2)
时, 我们使用了两个 CPU,所以两个 goroutine 可以一起被执行,以同样的频率交替打印 0 和 1。
runtime 包中的其它函数
中文文档在这里:https://studygolang.com/pkgdoc
这里就简单列举一下一些函数以及功能。
func GOROOT() string
GOROOT 返回 Go 的根目录。如果存在 GOROOT 环境变量,返回该变量的值;否则,返回创建 Go 时的根目录。
func Version() string
返回 Go 的版本字符串。它要么是递交的 hash 和创建时的日期;要么是发行标签如 "go1.3"。
func NumCPU() int
NumCPU返回本地机器的逻辑CPU个数(真 · 八核)。
func GC()
GC执行一次垃圾回收。(如果你迫切的希望做一次垃圾回收,可以调用此函数)
来源:https://www.cnblogs.com/lpgit/p/13430858.html


猜你喜欢
- 在待测试的私有函数所在的包内,新建一个xx_test.go文件书写方式如下:import ( "github.com/stretc
- 快速排序由于排序效率在同为O(N*logN)的几种排序方法中效率较高,因此经常被采用。该方法的基本思想是:1.先从数列中取出一个数作为基准数
- 物体跟踪效果展示 过程:一、初始化def Motor_Init(): global L_Motor, R
- MySQL ALTER语法如下:ALTER [IGNORE] TABLE tbl_name alter_spec [, alter_spec
- 本文实例讲述了GO语言Defer用法。分享给大家供大家参考。具体分析如下:defer:调用一个被 defer 的函数时在函数刚要返回之前延迟
- 本文实例讲述了Python获取网页上图片下载地址的方法。分享给大家供大家参考。具体如下:这里获取网页上图片的下载地址是正在写的数据采集中的一
- b 和 i 标签在现在的 Web 标准潜规则中是不推荐使用,甚至是反对使用的,因为认为他们只是“表现”粗体和斜体,而没有任何“语义”。更多的
- 前言本文从代码复用的角度一步一步演示如何从python普通代码进化到面向对象,并通过代码去解释一些面向对象的理论。所以,本文前面的内容都是非
- 实现功能excel表格中有4列数,分别为RMF计算得到的 β,γ,势能面及组态,需要挑选出相同 β 值下势能面最低时的组态。为了减小数据量,
- Exec 是 os 包中的一个子包,它可用于使用 Go 运行外部命令。Go exec 命令教程展示了如何在 Golang 中执行 shell
- 很多朋友对FrontPage2003中增加的网页布局功能很感兴趣,现在我们一起来深入了解这一实用功能。用FrontPage2003的“布局表
- 一、引言网络上充满了窃听,我们的信息很容易被不怀好意的人获得,给我们造成不好的影响。如果你需要在网络上传输机密或者敏感的隐私信息,为了防备别
- 一次性选中并修改多个相同的变量在编码的时候,有时候要批量替换一个变量的名字,但是又不想使用批量替换,因为在不同的作用域中是可以使用相同的变量
- 旁站查询来源:http://dns.aizhan.comhttp://s.tool.chinaz.com/samehttp://i.link
- A朋友:你直接问。B朋友:你女朋友睡了后,偷偷拿电脑查。C朋友:咱们不吊死在一棵树上,不对劲就早点脱身。D朋友:我觉得A的不行,直接问容易打
- 问题:MySQL 5.0不能使用自动增加字段。具体原因:引起的原因是在MySQL5.0.2版本中的The Server SQL Mode 中
- 查询数据库SELECT * FROM `student`查询结果idnamebirthday1张三1970-10-012李四1990-10-
- 在计算机软件领域,缓存(Cache)指的是将部分数据存储在内存中,以便下次能够更快地访问这些数据,这也是一个典型的用空间换时间的例子。一般用
- 看代码吧~# 加载库import pandas as pd# 데이터프레임을 만듭니다.dataframe = pd.DataFrame()
- 一、连接MYSQL。 格式: mysql -h主机地址 -u用户名 -p用户密码&nbs