goroutine是什么

goroutine即协程,使用go关键字开启一个协程异步执行代码。

注意,main函数也是个goroutine。

基本使用

使用go执行子任务,会交替执行(和时间片一样)。

主goroutine退出后,其它的工作goroutine也会自动退出(有点父子进程的感觉):

package main

import (
"fmt"
"time"
) func newTask() {
i := 0
for {
i++
fmt.Printf("new goroutine: i = %d\n", i)
time.Sleep(1 * time.Second) //延时1s
}
} func main() {
//创建一个 goroutine,启动另外一个任务
go newTask() i := 0
//main goroutine 循环打印
for {
i++
fmt.Printf("main goroutine: i = %d\n", i)
time.Sleep(1 * time.Second) //延时1s
}
//这里是加入了死循环,如果去掉,则程序会直接退出。
}

多个协程的顺序是不一定的。

var _ = runtime.GOMAXPROCS(3)
var a, b int
func u1() {
a = 1
b = 2
}
func u2() {
a = 3
b = 4
}
func p() {
println(a)
println(b)
}
func main() {
go u1() // 多个 goroutine 的执行顺序不定
go u2()
go p()
time.Sleep(1 * time.Second)
}

runtime包

Gosched

runtime.Gosched() //让别人先执行,需要同时需要时间片的时候才会有效,对方如果已经停了就还是自己执行。

就像孔融让梨(梨就是CPU时间片),A遇到runtime.Gosched()就先给B吃(让出时间片),但是如果B已经吃完了(B已经不需要时间片了),A就开始吃(A则开始占用CPU)。

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() //import "runtime"
/*
屏蔽runtime.Gosched()运行结果如下:
hello
hello 没有runtime.Gosched()运行结果如下:
world
world
hello
hello
*/
fmt.Println("hello")
}
}

优先调度:

你的程序可能出现一个 goroutine 在运行时阻止了其他 goroutine 的运行,比如程序中有一个不让调度器运行的 for 循环:

调度器会在 GC、Go 声明、阻塞 channel、阻塞系统调用和锁操作后再执行,也会在非内联函数调用时执行:

func main() {
done := false
go func() {
done = true
}()
//这里占用了调度,协程无法启动
for !done {
println("not done !") // 并不内联执行
}
println("done !")
} //可以添加 -m 参数来分析 for 代码块中调用的内联函数 修改: func main() {
done := false
go func() {
done = true
}()
for !done {
runtime.Gosched()
}
println("done !")
}

Goexit

runtime.Goexit() //将立即终止当前 goroutine 执⾏,调度器确保所有已注册 defer延迟调用被执行。

package main

import (
"fmt"
"runtime"
"time"
) //调用 runtime.Goexit() 将立即终止当前 goroutine 执⾏,调度器确保所有已注册 defer延迟调用被执行。 func main() {
go func() {
defer fmt.Println("A.defer") func() {
defer fmt.Println("B.defer")
runtime.Goexit() // 终止当前 goroutine, import "runtime"
fmt.Println("B") // 不会执行
}() defer fmt.Println("C.defer") //还没来得及注册,不会执行 fmt.Println("A") // 不会执行
}() //别忘了() //死循环,目的不让主goroutine结束
for {
time.Sleep(1 * time.Second)
}
} //执行结果:
//B.defer
//A.defer

GOMAXPROCS

调用 runtime.GOMAXPROCS() 用来设置可以并行计算的CPU核数的最大值,并返回之前的值。

示例代码:

func main() {
//n := runtime.GOMAXPROCS(1)
//打印结果:111111111111111111110000000000000000000011111...
n := runtime.GOMAXPROCS(2)
//打印结果:010101010101010101011001100101011010010100110...
fmt.Printf("n = %d\n", n) for {
go fmt.Print(0)
fmt.Print(1)
}
}

在第一次执行(runtime.GOMAXPROCS(1))时,最多同时只能有一个goroutine被执行。所以会打印很多1。

过了一段时间后,GO调度器会将其置为休眠,并唤醒另一个goroutine,这时候就开始打印很多0了,在打印的时候,goroutine是被调度到操作系统线程上的。

在第二次执行(runtime.GOMAXPROCS(2))时,我们使用了两个CPU,所以两个goroutine可以一起被执行,以同样的频率交替打印0和1。

最新文章

  1. net-force.nl/steganography writeup
  2. tornado 学习笔记17 HTTPServerRequest分析
  3. HTML5全局属性和事件详解
  4. HoverTree系统源码介绍
  5. bzoj2006 noi2010 超级钢琴 主席树 + 优先队列
  6. mysqli连接数据库常见函数
  7. Socket get http request
  8. C语言里的文件函数
  9. python学习笔记——列表生成式与生成器
  10. JAVA 统计字符串中中文,英文,数字,空格的个数
  11. UPX3.03+UpolyX.5 Shell v1.0 汉化绿色版
  12. SSO框架介绍前篇
  13. 【原创】最新的&quot;答题热&quot;中,前端的一点小工作
  14. Algorithm --&gt; 小于N的正整数含有1的个数
  15. 在原有数据库中使用 CodeFirst
  16. Maven之pom.xml配置文件详解
  17. MySQL性能测试工具sysbench的安装和使用
  18. [转载] Spring框架——AOP前置、后置、环绕、异常通知
  19. Hadoop HBase概念学习系列之RowKey设计(二十九)
  20. 说一说Servlet的生命周期

热门文章

  1. [Polkadot] 波卡链学习笔记
  2. 【SQL进阶】【分步写、联合各自排序、TIMESTAMPDIFF时间比较】Day04:多表查询
  3. 【Java EE】Day11 BootStrap、响应式布局、栅格系统、CSS样式、案例
  4. this关键字在JAVA和JS中的异同
  5. ATM购物车项目总结
  6. STM32与PS2的无线通信和相关函数介绍
  7. 基于K-means聚类算法进行客户人群分析
  8. 12、HSSFWorkbook实现多张sheet导出
  9. Django(2) - Django模板
  10. MongoDB从入门到实战之Docker快速安装MongoDB