0

0

Goroutines 的工作原理及主进程结束后如何处理

花韻仙語

花韻仙語

发布时间:2025-10-08 11:03:11

|

859人浏览过

|

来源于php中文网

原创

goroutines 的工作原理及主进程结束后如何处理

本文深入探讨了 Go 语言中 Goroutines 的工作机制,包括 Goroutines 的生命周期以及主进程结束后 Goroutines 的处理方式。通过分析一个向 MongoDB 插入大量数据的并发示例,解释了如何使用 sync.WaitGroup 来确保所有 Goroutines 完成后再退出程序。同时,提供了一个精简的可运行示例,帮助读者理解 Goroutines 的基本用法,并指导读者逐步构建更复杂的并发程序。

Goroutines 的基本概念

Goroutines 是 Go 语言中实现并发的核心机制。它们本质上是轻量级的线程,由 Go 运行时环境(runtime)进行管理。与传统的操作系统线程相比,Goroutines 的创建和销毁开销更小,上下文切换速度更快,因此可以轻松地创建成千上万个 Goroutines,从而实现高并发。

在 Go 语言中,启动一个 Goroutine 非常简单,只需要在函数调用前加上 go 关键字即可。例如:

go myFunction()

这将会创建一个新的 Goroutine 并并发执行 myFunction 函数。

Goroutines 的生命周期

Goroutines 的生命周期从创建开始,到函数执行完毕或发生 panic 结束。需要特别注意的是,当 main 函数返回时,程序会立即退出,而不会等待其他 Goroutines 完成。 这也是导致并发程序出现问题的常见原因。

使用 sync.WaitGroup 管理 Goroutines

为了确保所有 Goroutines 在 main 函数退出前完成,可以使用 sync.WaitGroup。 sync.WaitGroup 提供了一种简单的机制来等待一组 Goroutines 完成。

sync.WaitGroup 的使用方法如下:

Synthesys
Synthesys

Synthesys是一家领先的AI虚拟媒体平台,用户只需点击几下鼠标就可以制作专业的AI画外音和AI视频

下载
  1. Add(delta int): 在启动 Goroutine 之前,调用 Add 方法,增加计数器的值,表示需要等待的 Goroutine 的数量。
  2. Done(): 在 Goroutine 完成后,调用 Done 方法,减少计数器的值。
  3. Wait(): 在 main 函数中,调用 Wait 方法,阻塞当前 Goroutine(通常是 main 函数),直到计数器的值为 0,表示所有 Goroutines 都已完成。

以下是一个使用 sync.WaitGroup 的示例:

package main

import (
    "fmt"
    "sync"
    "time"
)

var waitGroup sync.WaitGroup

func worker(id int) {
    defer waitGroup.Done() // 确保 Goroutine 结束后调用 Done()
    fmt.Printf("Worker %d starting\n", id)
    time.Sleep(time.Second) // 模拟耗时操作
    fmt.Printf("Worker %d done\n", id)
}

func main() {
    for i := 1; i <= 3; i++ {
        waitGroup.Add(1) // 启动一个 Goroutine 前,增加计数器
        go worker(i)
    }

    waitGroup.Wait() // 等待所有 Goroutines 完成
    fmt.Println("All workers done")
}

在这个例子中,我们启动了 3 个 Goroutines 来执行 worker 函数。waitGroup.Add(1) 在每次启动 Goroutine 之前将计数器加 1,waitGroup.Done() 在每个 Goroutine 结束后将计数器减 1。waitGroup.Wait() 会阻塞 main 函数,直到计数器的值为 0,即所有 Goroutines 都已完成。

并发插入 MongoDB 的示例分析

以下是一个向 MongoDB 并发插入数据的示例(基于原问题中的代码进行简化和修正):

package main

import (
    "fmt"
    "labix.org/v2/mgo"
    "strconv"
    "sync"
    "time"
)

// Reading 结构体
type Reading struct {
    Id   string
    Name string
}

var waitGroup sync.WaitGroup

func main() {
    startTime := time.Now()

    // 连接 MongoDB
    session, err := mgo.Dial("localhost")
    if err != nil {
        panic(err)
    }
    defer session.Close()

    collection := session.DB("test").C("readings")

    readings := prepareReadings()
    fmt.Println("readings prepared: " + strconv.FormatFloat(time.Since(startTime).Seconds(), 'f', 2, 64))

    // 并发插入数据
    numReadings := 1000000
    for i := 1; i <= numReadings; i++ {
        waitGroup.Add(1)
        go insertReading(collection, readings)

        if i%100000 == 0 {
            fmt.Println("100000 readings queued for insert: " + strconv.FormatFloat(time.Since(startTime).Seconds(), 'f', 2, 64))
        }
    }
    waitGroup.Wait()

    fmt.Println("all readings inserted: " + strconv.FormatFloat(time.Since(startTime).Seconds(), 'f', 2, 64))
}

func insertReading(collection *mgo.Collection, readings []Reading) {
    defer waitGroup.Done() // 确保 Goroutine 结束后调用 Done()
    err := collection.Insert(readings...) // 插入 readings 切片中的所有元素

    if err != nil {
        fmt.Println("error insertReadings:", err)
    }
}

func prepareReadings() []Reading {
    var readings []Reading
    for i := 1; i <= 10; i++ { // 创建 10 个 Reading 对象
        readings = append(readings, Reading{Name: "Thing " + strconv.Itoa(i)})
    }
    return readings
}

注意事项:

  • 连接复用: 在并发环境下,尽量复用 MongoDB 连接,避免频繁创建和销毁连接,以提高性能。可以将 mgo.Session 对象传递给 Goroutines,或者使用连接池。
  • 错误处理: 在 Goroutines 中进行错误处理非常重要。需要检查 MongoDB 操作是否成功,并记录或处理错误。
  • 批量插入: 为了提高插入效率,可以考虑使用 MongoDB 的批量插入功能,一次性插入多个文档。
  • 数据竞争: 如果多个 Goroutines 同时访问和修改共享数据,需要使用互斥锁(sync.Mutex)或其他同步机制来避免数据竞争。

总结

Goroutines 是 Go 语言强大的并发特性,可以轻松地构建高性能的并发应用程序。理解 Goroutines 的生命周期以及如何使用 sync.WaitGroup 来管理 Goroutines 是编写并发程序的基础。在实际应用中,还需要注意连接复用、错误处理、批量插入和数据竞争等问题,以确保程序的正确性和性能。通过逐步构建和测试,可以更好地理解和掌握 Goroutines 的使用。

相关专题

更多
session失效的原因
session失效的原因

session失效的原因有会话超时、会话数量限制、会话完整性检查、服务器重启、浏览器或设备问题等等。详细介绍:1、会话超时:服务器为Session设置了一个默认的超时时间,当用户在一段时间内没有与服务器交互时,Session将自动失效;2、会话数量限制:服务器为每个用户的Session数量设置了一个限制,当用户创建的Session数量超过这个限制时,最新的会覆盖最早的等等。

314

2023.10.17

session失效解决方法
session失效解决方法

session失效通常是由于 session 的生存时间过期或者服务器关闭导致的。其解决办法:1、延长session的生存时间;2、使用持久化存储;3、使用cookie;4、异步更新session;5、使用会话管理中间件。

745

2023.10.18

cookie与session的区别
cookie与session的区别

本专题整合了cookie与session的区别和使用方法等相关内容,阅读专题下面的文章了解更详细的内容。

88

2025.08.19

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

338

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

542

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

53

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

197

2025.08.29

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

482

2023.08.10

C++ 高级模板编程与元编程
C++ 高级模板编程与元编程

本专题深入讲解 C++ 中的高级模板编程与元编程技术,涵盖模板特化、SFINAE、模板递归、类型萃取、编译时常量与计算、C++17 的折叠表达式与变长模板参数等。通过多个实际示例,帮助开发者掌握 如何利用 C++ 模板机制编写高效、可扩展的通用代码,并提升代码的灵活性与性能。

4

2026.01.23

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Go 教程
Go 教程

共32课时 | 4.1万人学习

Go语言实战之 GraphQL
Go语言实战之 GraphQL

共10课时 | 0.8万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号