Golang 并發(fā)編程入門Goroutine 簡介與基礎用法小結
一、什么是 Goroutine?
Goroutine 是 Golang 中的一種輕量級線程,用于實現(xiàn)并發(fā)操作。與傳統(tǒng)線程相比,Goroutine 的優(yōu)勢在于它具有更低的資源消耗和更高的效率。每個 Goroutine 在運行時被 Go 的調(diào)度器(Scheduler)管理,并且它們共享內(nèi)存空間。這使得在單個系統(tǒng)線程上運行成千上萬個 Goroutine 成為可能。
1、Goroutine 的特點:
- 輕量級:內(nèi)存開銷非常小,大約只需 2KB。
- 非搶占式調(diào)度:Goroutine 自愿讓出 CPU,通過協(xié)作完成任務切換。
- 并發(fā)與并行:Goroutine 提供并發(fā)模型,多個任務可以同時進行。
- 阻塞模型簡化開發(fā):避免了復雜的回調(diào)函數(shù),使代碼結構更加簡潔。
二、如何啟動 Goroutine
要啟動 Goroutine,我們只需要在函數(shù)調(diào)用前加上 go 關鍵字。此操作會將該函數(shù)在單獨的 Goroutine 中異步執(zhí)行,主程序不會等待它完成。
語法:
go 函數(shù)名(參數(shù))
示例 1:基本的 Goroutine 用法
package main
import (
"fmt"
"time"
)
func printMessage() {
fmt.Println("Goroutine is running!")
}
func main() {
go printMessage() // 啟動 Goroutine
time.Sleep(1 * time.Second) // 暫停主線程,確保 Goroutine 執(zhí)行完
fmt.Println("Main function is done.")
}輸出:
Goroutine is running!
Main function is done.
解釋:
- 主 Goroutine(即
main函數(shù))立即啟動了一個子 Goroutine 來運行printMessage函數(shù)。 - 如果沒有
time.Sleep,主 Goroutine 可能會在子 Goroutine 還未執(zhí)行時結束,導致沒有輸出。
三、主 Goroutine 與子 Goroutine 的關系
- 主 Goroutine:
main函數(shù)所在的 Goroutine,程序從這里開始運行。 - 子 Goroutine:由主 Goroutine 或其他 Goroutine 創(chuàng)建的 Goroutine。
關鍵點:如果主 Goroutine 結束,所有未完成的子 Goroutine 也會被強制終止。
示例 2:主線程結束導致子 Goroutine 終止
package main
import (
"fmt"
"time"
)
func main() {
go func() {
time.Sleep(2 * time.Second)
fmt.Println("This message may never appear.")
}()
fmt.Println("Main function is done.")
// 主 Goroutine 立即結束,子 Goroutine 沒有機會完成
}輸出:
Main function is done.
解釋:
主 Goroutine 結束時,所有未完成的子 Goroutine 會立即停止,因此不會看到子 Goroutine 的輸出。
四、Goroutine 的參數(shù)傳遞
在 Golang 中,Goroutine 支持傳遞參數(shù),但需要注意是按值傳遞,而不是按引用。
示例 3:Goroutine 傳遞參數(shù)
package main
import (
"fmt"
)
func printNumber(num int) {
fmt.Println("Number:", num)
}
func main() {
for i := 1; i <= 3; i++ {
go printNumber(i) // 啟動 Goroutine 傳遞參數(shù)
}
fmt.Scanln() // 等待用戶輸入,防止程序提前結束
}輸出(可能):
Number: 1
Number: 2
Number: 3
五、匿名函數(shù)中的變量捕獲問題
Goroutine 常與匿名函數(shù)一起使用,但要小心變量捕獲問題。在循環(huán)中啟動 Goroutine 時,匿名函數(shù)可能捕獲的不是當前迭代變量的值,而是循環(huán)結束后的變量。
示例 4:錯誤的變量捕獲
package main
import (
"fmt"
"time"
)
func main() {
for i := 1; i <= 3; i++ {
go func() {
fmt.Println(i) // 捕獲的可能是循環(huán)結束后的 i
}()
}
time.Sleep(1 * time.Second)
}輸出:
4
4
4
修改后的代碼:將變量作為參數(shù)傳遞
go func(n int) {
fmt.Println(n)
}(i)六、使用 WaitGroup 等待 Goroutine 完成
time.Sleep 不是等待 Goroutine 完成的最佳方式。Go 提供了 sync.WaitGroup 來管理多個 Goroutine 的同步。
示例 5:使用 WaitGroup
package main
import (
"fmt"
"sync"
)
func printMessage(msg string, wg *sync.WaitGroup) {
defer wg.Done() // Goroutine 完成時調(diào)用 Done()
fmt.Println(msg)
}
func main() {
var wg sync.WaitGroup
wg.Add(3) // 等待 3 個 Goroutine
go printMessage("Hello", &wg)
go printMessage("from", &wg)
go printMessage("Goroutine!", &wg)
wg.Wait() // 等待所有 Goroutine 完成
fmt.Println("All Goroutines are done.")
}輸出:
Hello
from
Goroutine!
All Goroutines are done.
解釋:
wg.Add(n)表示需要等待 n 個 Goroutine 完成。- 每個 Goroutine 結束時調(diào)用
wg.Done()。 wg.Wait()會阻塞主 Goroutine,直到所有任務完成。
七、Goroutine 的典型應用場景
- I/O 并發(fā):處理大量網(wǎng)絡請求,減少響應時間。
- 后臺任務:執(zhí)行定時任務、日志記錄等。
- 并行計算:分布計算任務,提高程序性能。
- 定時器與延時操作:如每隔一段時間執(zhí)行某個操作。
八、注意事項與最佳實踐
- 避免死鎖:當 Goroutine 等待彼此時可能出現(xiàn)死鎖,應小心處理共享資源。
- 合理使用 WaitGroup 和 Channel:確保 Goroutine 的同步與通信。
- 避免啟動過多 Goroutine:雖然 Goroutine 輕量,但過多的 Goroutine 也會占用系統(tǒng)資源。
- 調(diào)試工具:使用
go tool trace和pprof進行性能調(diào)優(yōu)和調(diào)試。
九、小結
Goroutine 是 Golang 的強大并發(fā)工具,其高效、易用的特點讓它成為開發(fā)者實現(xiàn)并發(fā)程序的首選。在這篇博客中,我們介紹了 Goroutine 的基礎用法、參數(shù)傳遞和同步機制,并演示了常見的應用場景和注意事項。
到此這篇關于Golang 并發(fā)編程入門:Goroutine 簡介與基礎用法的文章就介紹到這了,更多相關goland goroutine用法內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
golang值類型轉(zhuǎn)換成[]uint8類型的操作
這篇文章主要介紹了golang值類型轉(zhuǎn)換成[]uint8類型的操作,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2021-05-05
GO語言實現(xiàn)的http抓包分析工具pproxy介紹
這篇文章主要介紹了GO語言實現(xiàn)的http抓包分析工具pproxy介紹,本文同時對比了Fiddler、Charles等抓包軟件,需要的朋友可以參考下2015-03-03
go語言中數(shù)據(jù)接口set集合的實現(xiàn)
set集合是一種常見的數(shù)據(jù)結構,它代表了一個唯一元素的集合,本文主要介紹了set的基本特性,包括唯一性、無序性、可變性和集合運算,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2024-10-10
教你用go語言實現(xiàn)比特幣交易功能(Transaction)
每一筆比特幣交易都會創(chuàng)造輸出,輸出都會被區(qū)塊鏈記錄下來。給某個人發(fā)送比特幣,實際上意味著創(chuàng)造新的 UTXO 并注冊到那個人的地址,可以為他所用,今天通過本文給大家分享go語言實現(xiàn)比特幣交易功能,一起看看吧2021-05-05
Golang創(chuàng)建第一個web項目(Gin+Gorm)
本文主要介紹了Golang創(chuàng)建第一個web項目(Gin+Gorm),文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2024-06-06

