Golang 并发编程入门Goroutine 简介与基础用法小结

 更新时间:2024年11月13日 12:05:08   作者:Linke-  
Goroutine 是 Golang 中的一种轻量级线程,用于实现并发操作,与传统线程相比,Goroutine 的优势在于它具有更低的资源消耗和更高的效率,本文介绍Golang 并发编程入门Goroutine 简介与基础用法小结,感兴趣的朋友一起看看吧

一、什么是 Goroutine?

Goroutine 是 Golang 中的一种轻量级线程,用于实现并发操作。与传统线程相比,Goroutine 的优势在于它具有更低的资源消耗和更高的效率。每个 Goroutine 在运行时被 Go 的调度器(Scheduler)管理,并且它们共享内存空间。这使得在单个系统线程上运行成千上万个 Goroutine 成为可能。

1、Goroutine 的特点:

  • 轻量级:内存开销非常小,大约只需 2KB。
  • 非抢占式调度:Goroutine 自愿让出 CPU,通过协作完成任务切换。
  • 并发与并行:Goroutine 提供并发模型,多个任务可以同时进行。
  • 阻塞模型简化开发:避免了复杂的回调函数,使代码结构更加简洁。

二、如何启动 Goroutine

要启动 Goroutine,我们只需要在函数调用前加上 go 关键字。此操作会将该函数在单独的 Goroutine 中异步执行,主程序不会等待它完成。

语法:

go 函数名(参数)

示例 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 执行完
    fmt.Println("Main function is done.")
}

输出:

Goroutine is running!  
Main function is done.

解释:

  • 主 Goroutine(即 main 函数)立即启动了一个子 Goroutine 来运行 printMessage 函数。
  • 如果没有 time.Sleep,主 Goroutine 可能会在子 Goroutine 还未执行时结束,导致没有输出。

三、主 Goroutine 与子 Goroutine 的关系

  • 主 Goroutinemain 函数所在的 Goroutine,程序从这里开始运行。
  • 子 Goroutine:由主 Goroutine 或其他 Goroutine 创建的 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 的参数传递

在 Golang 中,Goroutine 支持传递参数,但需要注意是按值传递,而不是按引用。

示例 3:Goroutine 传递参数

package main
import (
    "fmt"
)
func printNumber(num int) {
    fmt.Println("Number:", num)
}
func main() {
    for i := 1; i <= 3; i++ {
        go printNumber(i) // 启动 Goroutine 传递参数
    }
    fmt.Scanln() // 等待用户输入,防止程序提前结束
}

输出(可能):

Number: 1  
Number: 2  
Number: 3

五、匿名函数中的变量捕获问题

Goroutine 常与匿名函数一起使用,但要小心变量捕获问题。在循环中启动 Goroutine 时,匿名函数可能捕获的不是当前迭代变量的值,而是循环结束后的变量。

示例 4:错误的变量捕获

package main
import (
    "fmt"
    "time"
)
func main() {
    for i := 1; i <= 3; i++ {
        go func() {
            fmt.Println(i) // 捕获的可能是循环结束后的 i
        }()
    }
    time.Sleep(1 * time.Second)
}

输出:

4  
4  
4

修改后的代码:将变量作为参数传递

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 完成时调用 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 结束时调用 wg.Done()
  • wg.Wait() 会阻塞主 Goroutine,直到所有任务完成。

七、Goroutine 的典型应用场景

  • I/O 并发:处理大量网络请求,减少响应时间。
  • 后台任务:执行定时任务、日志记录等。
  • 并行计算:分布计算任务,提高程序性能。
  • 定时器与延时操作:如每隔一段时间执行某个操作。

八、注意事项与最佳实践

  • 避免死锁:当 Goroutine 等待彼此时可能出现死锁,应小心处理共享资源。
  • 合理使用 WaitGroup 和 Channel:确保 Goroutine 的同步与通信。
  • 避免启动过多 Goroutine:虽然 Goroutine 轻量,但过多的 Goroutine 也会占用系统资源。
  • 调试工具:使用 go tool tracepprof 进行性能调优和调试。

九、小结

Goroutine 是 Golang 的强大并发工具,其高效、易用的特点让它成为开发者实现并发程序的首选。在这篇博客中,我们介绍了 Goroutine 的基础用法、参数传递和同步机制,并演示了常见的应用场景和注意事项。

到此这篇关于Golang 并发编程入门:Goroutine 简介与基础用法的文章就介绍到这了,更多相关goland goroutine用法内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Go语言常见错误之any没传递任何信息解决分析

    Go语言常见错误之any没传递任何信息解决分析

    Go语言,由于其高效强大的并行处理能力和优雅简单的设计哲学,一直以来都是编程世界的宠儿,然而,对于一些Go新手和甚至熟悉Go的程序员也可能会遇到一个常见的错误: any没传递任何信息,那么,如何规避这个错误,本文将揭示其中的秘密
    2024-01-01
  • golang值类型转换成[]uint8类型的操作

    golang值类型转换成[]uint8类型的操作

    这篇文章主要介绍了golang值类型转换成[]uint8类型的操作,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2021-05-05
  • GO语言实现的http抓包分析工具pproxy介绍

    GO语言实现的http抓包分析工具pproxy介绍

    这篇文章主要介绍了GO语言实现的http抓包分析工具pproxy介绍,本文同时对比了Fiddler、Charles等抓包软件,需要的朋友可以参考下
    2015-03-03
  • Go语言输出函数使用详解

    Go语言输出函数使用详解

    这篇文章主要介绍了Go语言输出函数使用详解的相关资料,需要的朋友可以参考下
    2023-08-08
  • go语言中数据接口set集合的实现

    go语言中数据接口set集合的实现

    set集合是一种常见的数据结构,它代表了一个唯一元素的集合,本文主要介绍了set的基本特性,包括唯一性、无序性、可变性和集合运算,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2024-10-10
  • 教你用go语言实现比特币交易功能(Transaction)

    教你用go语言实现比特币交易功能(Transaction)

    每一笔比特币交易都会创造输出,输出都会被区块链记录下来。给某个人发送比特币,实际上意味着创造新的 UTXO 并注册到那个人的地址,可以为他所用,今天通过本文给大家分享go语言实现比特币交易功能,一起看看吧
    2021-05-05
  • Golang创建第一个web项目(Gin+Gorm)

    Golang创建第一个web项目(Gin+Gorm)

    本文主要介绍了Golang创建第一个web项目(Gin+Gorm),文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2024-06-06
  • golang结构化日志slog的用法简介

    golang结构化日志slog的用法简介

    日志是任何软件的重要组成部分,Go 提供了一个内置日志包(slog),在本文中,小编将简单介绍一下slog包的功能以及如何在 Go 应用程序中使用它,感兴趣的可以了解下
    2023-09-09
  • 详解Golang中日志库glog的使用

    详解Golang中日志库glog的使用

    golang/glog 是 C++ 版本 google/glog 的 Go 版本实现,基本实现了原生 glog 的日志格式,下面大家就跟随小编一起了解一下glog的具体使用吧
    2023-09-09
  • 详解Golang如何使用Debug库优化代码

    详解Golang如何使用Debug库优化代码

    这篇文章将针对Golang的debug库进行全面解读,涵盖其核心组件、高级功能和实战技巧,文中的示例代码讲解详细,有需要的小伙伴可以参考下
    2024-02-02

最新评论