Goroutines和线程

摘自:gopl 9.8. Goroutines和线程


 

在上一章中我们说goroutine和操作系统的线程区别可以先忽略。尽管两者的区别实际上只是 一个量的区别,但量变会引起质变的道理同样适用于goroutine和线程。现在正是我们来区分 开两者的最佳时机。

9.8.1. 动态栈

每一个OS线程都有一个固定大小的内存块(一般会是2MB)来做栈,这个栈会用来存储当前正 在被调用或挂起(指在调用其它函数时)的函数的内部变量。这个固定大小的栈同时很大又很 小。因为2MB的栈对于一个小小的goroutine来说是很大的内存浪费,比如对于我们用到的, 一个只是用来WaitGroup之后关闭channel的goroutine来说。而对于go程序来说,同时创建成 百上千个goroutine是非常普遍的,如果每一个goroutine都需要这么大的栈的话,那这么多的 goroutine就不太可能了。除去大小的问题之外,固定大小的栈对于更复杂或者更深层次的递 归函数调用来说显然是不够的。修改固定的大小可以提升空间的利用率允许创建更多的线 程,并且可以允许更深的递归调用,不过这两者是没法同时兼备的。 相反,一个goroutine会以一个很小的栈开始其生命周期,一般只需要2KB。一个goroutine的 栈,和操作系统线程一样,会保存其活跃或挂起的函数调用的本地变量,但是和OS线程不太 一样的是一个goroutine的栈大小并不是固定的;栈的大小会根据需要动态地伸缩。而 goroutine的栈的最大值有1GB,比传统的固定大小的线程栈要大得多,尽管一般情况下,大 多goroutine都不需要这么大的栈。 

9.8.2. Goroutine调度

OS线程会被操作系统内核调度。每几毫秒,一个硬件计时器会中断处理器,这会调用一个叫 作scheduler的内核函数。这个函数会挂起当前执行的线程并保存内存中它的寄存器内容,检 查线程列表并决定下一次哪个线程可以被运行,并从内存中恢复该线程的寄存器信息,然后 恢复执行该线程的现场并开始执行线程。因为操作系统线程是被内核所调度,所以从一个线 程向另一个“移动”需要完整的上下文切换,也就是说,保存一个用户线程的状态到内存,恢复 另一个线程的到寄存器,然后更新调度器的数据结构。这几步操作很慢,因为其局部性很差 需要几次内存访问,并且会增加运行的cpu周期。

Go的运行时包含了其自己的调度器,这个调度器使用了一些技术手段,比如m:n调度,因为 其会在n个操作系统线程上多工(调度)m个goroutine。Go调度器的工作和内核的调度是相似 的,但是这个调度器只关注单独的Go程序中的goroutine(译注:按程序独立)。 gopl Goroutines和线程 368 和操作系统的线程调度不同的是,Go调度器并不是用一个硬件定时器而是被Go语言"建筑"本 身进行调度的。例如当一个goroutine调用了time.Sleep或者被channel调用或者mutex操作阻 塞时,调度器会使其进入休眠并开始执行另一个goroutine直到时机到了再去唤醒第一个 goroutine。因为这种调度方式不需要进入内核的上下文,所以重新调度一个goroutine比调度 一个线程代价要低得多。 练习 9.5: 写一个有两个goroutine的程序,两个goroutine会向两个无buffer channel反复地发送 ping-pong消息。这样的程序每秒可以支持多少次通信?

9.8.3. GOMAXPROCS

Go的调度器使用了一个叫做GOMAXPROCS的变量来决定会有多少个操作系统的线程同时执 行Go的代码。其默认的值是运行机器上的CPU的核心数,所以在一个有8个核心的机器上 时,调度器一次会在8个OS线程上去调度GO代码。(GOMAXPROCS是前面说的m:n调度中的 n)。在休眠中的或者在通信中被阻塞的goroutine是不需要一个对应的线程来做调度的。在I/O 中或系统调用中或调用非Go语言函数时,是需要一个对应的操作系统线程的,但是 GOMAXPROCS并不需要将这几种情况计算在内。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值