make loop

This commit is contained in:
chai2010 2016-01-08 21:06:07 +08:00
parent 93076843ad
commit 8c235ad220

View File

@ -1,9 +1,9 @@
### 9.8.2. Goroutine調度 ### 9.8.2. Goroutine調度
OS线程会被操作系统内核调度。每几毫秒一个硬件计时器会中断处理器这会调用一个叫作scheduler的内核函数。这个函数会挂起当前执行的线程并保存内存中它的寄存器内容检查线程列表并决定下一次哪个线程可以被运行并从内存中恢复该线程的寄存器信息然后恢复执行该线程的现场并开始执行线程。因为操作系统线程是被内核所调度所以从一个线程向另一个“移动”需要完整的上下文切换也就是说保存一个用户线程的状态到内存恢复另一个线程的到寄存器然后更新调度器的数据结构。这几步操作很慢因为其局部性很差需要几次内存访问并且会增加运行的cpu周期。 OS線程會被操作繫統內核調度。每幾毫秒一個硬件計時器會中斷處理器這會調用一個叫作scheduler的內核函數。這個函數會掛起當前執行的線程併保存內存中它的寄存器內容檢査線程列表併決定下一次哪個線程可以被運行併從內存中恢複該線程的寄存器信息然後恢複執行該線程的現場併開始執行線程。因爲操作繫統線程是被內核所調度所以從一個線程向另一個“移動”需要完整的上下文切換也就是説保存一個用戶線程的狀態到內存恢複另一個線程的到寄存器然後更新調度器的數據結構。這幾步操作很慢因爲其局部性很差需要幾次內存訪問併且會增加運行的cpu週期。
Go的运行时包含了其自己的调度器这个调度器使用了一些技术手段比如m:n调度因为其会在n个操作系统线程上多工(调度)m个goroutine。Go调度器的工作和内核的调度是相似的但是这个调度器只关注单独的Go程序中的goroutine(译注:按程序独立)。 Go的運行時包含了其自己的調度器這個調度器使用了一些技術手段比如m:n調度因爲其會在n個操作繫統線程上多工(調度)m個goroutine。Go調度器的工作和內核的調度是相似的但是這個調度器隻關註單獨的Go程序中的goroutine(譯註:按程序獨立)。
和操作系统的线程调度不同的是Go调度器并不是用一个硬件定时器而是被Go语言"建筑"本身进行调度的。例如当一个goroutine调用了time.Sleep或者被channel调用或者mutex操作阻塞时调度器会使其进入休眠并开始执行另一个goroutine直到时机到了再去唤醒第一个goroutine。因为因为这种调度方式不需要进入内核的上下文所以重新调度一个goroutine比调度一个线程代价要低得多。 和操作繫統的線程調度不同的是Go調度器併不是用一個硬件定時器而是被Go語言"建築"本身進行調度的。例如當一個goroutine調用了time.Sleep或者被channel調用或者mutex操作阻塞時調度器會使其進入休眠併開始執行另一個goroutine直到時機到了再去喚醒第一個goroutine。因爲因爲這種調度方式不需要進入內核的上下文所以重新調度一個goroutine比調度一個線程代價要低得多。
练习 9.5: 写一个有两个goroutine的程序两个goroutine会向两个无buffer channel反复地发送ping-pong消息。这样的程序每秒可以支持多少次通信? 練習 9.5: 寫一個有兩個goroutine的程序兩個goroutine會向兩個無buffer channel反複地發送ping-pong消息。這樣的程序每秒可以支持多少次通信?