设为首页收藏本站
网站公告 | 这是第一条公告
     

 找回密码
 立即注册
缓存时间02 现在时间02 缓存数据 我们最后一次见面是一起看《摆渡人》里面有句台词:原来此刻坐在身边的人也会走散。他紧抓了我手一下,我莫名心里一沉。过了一周跟我说了分手。但是那个手的力度我还记得。

我们最后一次见面是一起看《摆渡人》里面有句台词:原来此刻坐在身边的人也会走散。他紧抓了我手一下,我莫名心里一沉。过了一周跟我说了分手。但是那个手的力度我还记得。 -- 空心

查看: 649|回复: 2

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

[复制链接]

  离线 

TA的专栏

等级头衔

等級:晓枫资讯-列兵

在线时间
0 小时

积分成就
威望
0
贡献
39
主题
27
精华
0
金钱
103
积分
66
注册时间
2023-10-3
最后登录
2025-6-1

发表于 2024-11-14 20:37:25 | 显示全部楼层 |阅读模式
目录


  • 一、什么是 Goroutine?

    • 1、Goroutine 的特点:

  • 二、如何启动 Goroutine
  • 三、主 Goroutine 与子 Goroutine 的关系
  • 四、Goroutine 的参数传递
  • 五、匿名函数中的变量捕获问题
  • 六、使用 WaitGroup 等待 Goroutine 完成
  • 七、Goroutine 的典型应用场景
  • 八、注意事项与最佳实践
  • 九、小结

一、什么是 Goroutine?

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

1、Goroutine 的特点:


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

二、如何启动 Goroutine

要启动 Goroutine,我们只需要在函数调用前加上
  1. go
复制代码
关键字。此操作会将该函数在单独的 Goroutine 中异步执行,主程序不会等待它完成。
语法:
  1. go 函数名(参数)
复制代码
示例 1:基本的 Goroutine 用法
  1. package main
  2. import (
  3.     "fmt"
  4.     "time"
  5. )
  6. func printMessage() {
  7.     fmt.Println("Goroutine is running!")
  8. }
  9. func main() {
  10.     go printMessage() // 启动 Goroutine
  11.     time.Sleep(1 * time.Second) // 暂停主线程,确保 Goroutine 执行完
  12.     fmt.Println("Main function is done.")
  13. }
复制代码
输出:
  1. Goroutine is running!  Main function is done.
复制代码
解释:

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

三、主 Goroutine 与子 Goroutine 的关系


  • 主 Goroutine
    1. main
    复制代码
    函数所在的 Goroutine,程序从这里开始运行。
  • 子 Goroutine:由主 Goroutine 或其他 Goroutine 创建的 Goroutine。
关键点:如果主 Goroutine 结束,所有未完成的子 Goroutine 也会被强制终止。
示例 2:主线程结束导致子 Goroutine 终止
  1. package main
  2. import (
  3.     "fmt"
  4.     "time"
  5. )
  6. func main() {
  7.     go func() {
  8.         time.Sleep(2 * time.Second)
  9.         fmt.Println("This message may never appear.")
  10.     }()
  11.     fmt.Println("Main function is done.")
  12.     // 主 Goroutine 立即结束,子 Goroutine 没有机会完成
  13. }
复制代码
输出:
  1. Main function is done.
复制代码
解释:
主 Goroutine 结束时,所有未完成的子 Goroutine 会立即停止,因此不会看到子 Goroutine 的输出。

四、Goroutine 的参数传递

在 Golang 中,Goroutine 支持传递参数,但需要注意是按值传递,而不是按引用。
示例 3:Goroutine 传递参数
  1. package main
  2. import (
  3.     "fmt"
  4. )
  5. func printNumber(num int) {
  6.     fmt.Println("Number:", num)
  7. }
  8. func main() {
  9.     for i := 1; i <= 3; i++ {
  10.         go printNumber(i) // 启动 Goroutine 传递参数
  11.     }
  12.     fmt.Scanln() // 等待用户输入,防止程序提前结束
  13. }
复制代码
输出(可能):
  1. Number: 1  Number: 2  Number: 3
复制代码
五、匿名函数中的变量捕获问题

Goroutine 常与匿名函数一起使用,但要小心变量捕获问题。在循环中启动 Goroutine 时,匿名函数可能捕获的不是当前迭代变量的值,而是循环结束后的变量。
示例 4:错误的变量捕获
  1. package main
  2. import (
  3.     "fmt"
  4.     "time"
  5. )
  6. func main() {
  7.     for i := 1; i <= 3; i++ {
  8.         go func() {
  9.             fmt.Println(i) // 捕获的可能是循环结束后的 i
  10.         }()
  11.     }
  12.     time.Sleep(1 * time.Second)
  13. }
复制代码
输出:
  1. 4  4  4
复制代码
修改后的代码:将变量作为参数传递
  1. go func(n int) {
  2.     fmt.Println(n)
  3. }(i)
复制代码
六、使用 WaitGroup 等待 Goroutine 完成
  1. time.Sleep
复制代码
不是等待 Goroutine 完成的最佳方式。Go 提供了 sync.WaitGroup 来管理多个 Goroutine 的同步。
示例 5:使用 WaitGroup
  1. package main
  2. import (
  3.     "fmt"
  4.     "sync"
  5. )
  6. func printMessage(msg string, wg *sync.WaitGroup) {
  7.     defer wg.Done() // Goroutine 完成时调用 Done()
  8.     fmt.Println(msg)
  9. }
  10. func main() {
  11.     var wg sync.WaitGroup
  12.     wg.Add(3) // 等待 3 个 Goroutine
  13.     go printMessage("Hello", &wg)
  14.     go printMessage("from", &wg)
  15.     go printMessage("Goroutine!", &wg)
  16.     wg.Wait() // 等待所有 Goroutine 完成
  17.     fmt.Println("All Goroutines are done.")
  18. }
复制代码
输出:
  1. Hello  from  Goroutine!  All Goroutines are done.
复制代码
解释:

    1. wg.Add(n)
    复制代码
    表示需要等待 n 个 Goroutine 完成。
  • 每个 Goroutine 结束时调用
    1. wg.Done()
    复制代码

    1. wg.Wait()
    复制代码
    会阻塞主 Goroutine,直到所有任务完成。

七、Goroutine 的典型应用场景


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

八、注意事项与最佳实践


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

九、小结

Goroutine 是 Golang 的强大并发工具,其高效、易用的特点让它成为开发者实现并发程序的首选。在这篇博客中,我们介绍了 Goroutine 的基础用法、参数传递和同步机制,并演示了常见的应用场景和注意事项。
到此这篇关于Golang 并发编程入门:Goroutine 简介与基础用法的文章就介绍到这了,更多相关goland goroutine用法内容请搜索晓枫资讯以前的文章或继续浏览下面的相关文章希望大家以后多多支持晓枫资讯!

免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!
晓枫资讯-科技资讯社区-免责声明
免责声明:以上内容为本网站转自其它媒体,相关信息仅为传递更多信息之目的,不代表本网观点,亦不代表本网站赞同其观点或证实其内容的真实性。
      1、注册用户在本社区发表、转载的任何作品仅代表其个人观点,不代表本社区认同其观点。
      2、管理员及版主有权在不事先通知或不经作者准许的情况下删除其在本社区所发表的文章。
      3、本社区的文章部分内容可能来源于网络,仅供大家学习与参考,如有侵权,举报反馈:点击这里给我发消息进行删除处理。
      4、本社区一切资源不代表本站立场,并不代表本站赞同其观点和对其真实性负责。
      5、以上声明内容的最终解释权归《晓枫资讯-科技资讯社区》所有。
http://bbs.yzwlo.com 晓枫资讯--游戏IT新闻资讯~~~

  离线 

TA的专栏

  • 打卡等级:无名新人
  • 打卡总天数:1
  • 打卡月天数:0
  • 打卡总奖励:10
  • 最近打卡:2025-02-19 13:32:10
等级头衔

等級:晓枫资讯-列兵

在线时间
0 小时

积分成就
威望
0
贡献
0
主题
0
精华
0
金钱
22
积分
4
注册时间
2024-12-6
最后登录
2025-2-19

发表于 2025-2-10 11:13:12 | 显示全部楼层
路过,支持一下
http://bbs.yzwlo.com 晓枫资讯--游戏IT新闻资讯~~~

  离线 

TA的专栏

等级头衔

等級:晓枫资讯-列兵

在线时间
0 小时

积分成就
威望
0
贡献
0
主题
0
精华
0
金钱
21
积分
22
注册时间
2022-12-26
最后登录
2022-12-26

发表于 2025-6-29 09:43:27 | 显示全部楼层
感谢楼主,顶。
http://bbs.yzwlo.com 晓枫资讯--游戏IT新闻资讯~~~
严禁发布广告,淫秽、色情、赌博、暴力、凶杀、恐怖、间谍及其他违反国家法律法规的内容。!晓枫资讯-社区
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

1楼
2楼
3楼

手机版|晓枫资讯--科技资讯社区 本站已运行

CopyRight © 2022-2025 晓枫资讯--科技资讯社区 ( BBS.yzwlo.com ) . All Rights Reserved .

晓枫资讯--科技资讯社区

本站内容由用户自主分享和转载自互联网,转载目的在于传递更多信息,并不代表本网赞同其观点和对其真实性负责。

如有侵权、违反国家法律政策行为,请联系我们,我们会第一时间及时清除和处理! 举报反馈邮箱:点击这里给我发消息

Powered by Discuz! X3.5

快速回复 返回顶部 返回列表