欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Golang 并發(fā)編程入門Goroutine 簡介與基礎用法小結

 更新時間:2024年11月13日 12:05:08   作者:Linke-  
Goroutine 是 Golang 中的一種輕量級線程,用于實現(xiàn)并發(fā)操作,與傳統(tǒng)線程相比,Goroutine 的優(yōu)勢在于它具有更低的資源消耗和更高的效率,本文介紹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 的關系

  • 主 Goroutinemain 函數(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 tracepprof 進行性能調(diào)優(yōu)和調(diào)試。

九、小結

Goroutine 是 Golang 的強大并發(fā)工具,其高效、易用的特點讓它成為開發(fā)者實現(xiàn)并發(fā)程序的首選。在這篇博客中,我們介紹了 Goroutine 的基礎用法、參數(shù)傳遞和同步機制,并演示了常見的應用場景和注意事項。

到此這篇關于Golang 并發(fā)編程入門:Goroutine 簡介與基礎用法的文章就介紹到這了,更多相關goland goroutine用法內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!

相關文章

  • Go語言常見錯誤之a(chǎn)ny沒傳遞任何信息解決分析

    Go語言常見錯誤之a(chǎn)ny沒傳遞任何信息解決分析

    Go語言,由于其高效強大的并行處理能力和優(yōu)雅簡單的設計哲學,一直以來都是編程世界的寵兒,然而,對于一些Go新手和甚至熟悉Go的程序員也可能會遇到一個常見的錯誤:?any沒傳遞任何信息,那么,如何規(guī)避這個錯誤,本文將揭示其中的秘密
    2024-01-01
  • golang值類型轉(zhuǎn)換成[]uint8類型的操作

    golang值類型轉(zhuǎn)換成[]uint8類型的操作

    這篇文章主要介紹了golang值類型轉(zhuǎn)換成[]uint8類型的操作,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2021-05-05
  • GO語言實現(xiàn)的http抓包分析工具pproxy介紹

    GO語言實現(xiàn)的http抓包分析工具pproxy介紹

    這篇文章主要介紹了GO語言實現(xiàn)的http抓包分析工具pproxy介紹,本文同時對比了Fiddler、Charles等抓包軟件,需要的朋友可以參考下
    2015-03-03
  • Go語言輸出函數(shù)使用詳解

    Go語言輸出函數(shù)使用詳解

    這篇文章主要介紹了Go語言輸出函數(shù)使用詳解的相關資料,需要的朋友可以參考下
    2023-08-08
  • go語言中數(shù)據(jù)接口set集合的實現(xiàn)

    go語言中數(shù)據(jù)接口set集合的實現(xiàn)

    set集合是一種常見的數(shù)據(jù)結構,它代表了一個唯一元素的集合,本文主要介紹了set的基本特性,包括唯一性、無序性、可變性和集合運算,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2024-10-10
  • 教你用go語言實現(xiàn)比特幣交易功能(Transaction)

    教你用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)

    本文主要介紹了Golang創(chuàng)建第一個web項目(Gin+Gorm),文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2024-06-06
  • golang結構化日志slog的用法簡介

    golang結構化日志slog的用法簡介

    日志是任何軟件的重要組成部分,Go?提供了一個內(nèi)置日志包(slog),在本文中,小編將簡單介紹一下slog包的功能以及如何在?Go?應用程序中使用它,感興趣的可以了解下
    2023-09-09
  • 詳解Golang中日志庫glog的使用

    詳解Golang中日志庫glog的使用

    golang/glog?是?C++?版本?google/glog?的?Go?版本實現(xiàn),基本實現(xiàn)了原生?glog?的日志格式,下面大家就跟隨小編一起了解一下glog的具體使用吧
    2023-09-09
  • 詳解Golang如何使用Debug庫優(yōu)化代碼

    詳解Golang如何使用Debug庫優(yōu)化代碼

    這篇文章將針對Golang的debug庫進行全面解讀,涵蓋其核心組件、高級功能和實戰(zhàn)技巧,文中的示例代碼講解詳細,有需要的小伙伴可以參考下
    2024-02-02

最新評論