Go语言使用Goroutine并发打印的项目实践

 更新时间:2025年08月14日 10:40:14   作者:程序员爱钓鱼  
本文通过并发打印任务案例,讲解Go语言Goroutine的使用、同步控制及高并发优势,文中通过示例代码介绍的非常详细,需要的朋友们下面随着小编来一起学习学习吧

在本篇文章中,我们将从最基础的 Goroutine 并发模型入手,逐步掌握 Go 语言中并发控制的核心思想。通过一个简单的“并发打印任务”案例,掌握 Goroutine 的使用、输出顺序的控制、主协程退出时机等关键知识。

一、为什么学习 Goroutine?

Goroutine 是 Go 语言最核心的并发原语。相比于线程,它更加轻量:

  • 启动成本极低(大约只需 2KB 内存)
  • 数量可以轻松达到上万甚至百万级
  • 使用非常简单,只需一个 go 关键字

Go 的并发理念是:不要通过共享内存来通信,而要通过通信来共享内存。

掌握 Goroutine,是理解 Go 并发模型的第一步。

二、实战目标

我们将实现一个小项目:

  1. 启动多个 Goroutine 并发打印任务信息
  2. 每个任务打印固定内容
  3. 确保主程序等待所有任务完成
  4. 引入 sync.WaitGroup 实现同步控制

三、基础版本:启动多个 Goroutine

下面是最简单的 Goroutine 并发打印实现:

package main

import (
    "fmt"
    "time"
)

func printTask(id int) {
    fmt.Printf("任务 %d 正在执行\n", id)
}

func main() {
    for i := 1; i <= 5; i++ {
        go printTask(i)
    }

    // 给协程一点时间完成(不推荐的方式)
    time.Sleep(time.Second)
    fmt.Println("主程序结束")
}

运行结果(示例):

任务 2 正在执行
任务 4 正在执行
任务 1 正在执行
任务 3 正在执行
任务 5 正在执行
主程序结束

你会注意到,输出顺序可能与代码中的顺序不同,这是并发执行的结果。

四、问题:主程序过早退出

在上面例子中我们使用了 time.Sleep 来“凑时间”,这是一种不可靠且危险的方式。如果任务执行时间更长,就会被主程序强制退出。

为了解决这个问题,我们引入 sync.WaitGroup

五、进阶版本:使用 WaitGroup 等待所有 Goroutine 完成

package main

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

// 模拟执行任务
func printTask(id int, wg *sync.WaitGroup) {
    defer wg.Done() // 标记一个任务完成
    fmt.Printf("任务 %d 开始执行\n", id)
    time.Sleep(time.Duration(id) * 200 * time.Millisecond) // 模拟不同任务耗时
    fmt.Printf("任务 %d 执行完毕\n", id)
}

func main() {
    var wg sync.WaitGroup

    taskCount := 5
    for i := 1; i <= taskCount; i++ {
        wg.Add(1) // 增加一个任务
        go printTask(i, &wg)
    }

    wg.Wait() // 等待所有任务完成
    fmt.Println("所有任务完成,主程序退出")
}

输出示例:

任务 1 开始执行
任务 2 开始执行
任务 3 开始执行
任务 4 开始执行
任务 5 开始执行
任务 1 执行完毕
任务 2 执行完毕
任务 3 执行完毕
任务 4 执行完毕
任务 5 执行完毕
所有任务完成,主程序退出

六、重点知识点解析

1.go关键字

只需在函数前加 go,即可将该函数交由调度器异步执行,立即返回

2.sync.WaitGroup

  • wg.Add(n):添加 n 个任务
  • wg.Done():在任务完成后调用
  • wg.Wait():阻塞主线程,直到所有任务完成

3.defer用法

使用 defer wg.Done() 可以确保即使函数中途返回也能正确减计数。

七、延伸思考

  • 若有 10万个任务要并发执行,Goroutine 是否能承载?(答案是:可以,但建议使用并发控制池)
  • 如果某个任务失败,是否能提前取消其他任务?(答案是:可以结合 context 实现)

八、真实应用场景举例

  • 批量调用多个接口(如:并发请求天气接口、发邮件)
  • 批量处理文件(压缩、上传、转码)
  • 并发爬虫(抓取多个页面)
  • 消息队列并发消费任务

九、小结

本文通过“并发打印任务”这一简单示例,向你展示了 Go 并发编程的基本能力:

  • 启动 Goroutine 的方法
  • 并发任务间的非确定性
  • 如何使用 WaitGroup 来保证主程序的正确退出
  • Goroutine 的轻量特性,适合高并发场景

并发是 Go 的灵魂,控制好协程的生命周期和调度能力,将直接影响系统的健壮性与性能。

到此这篇关于Go语言使用Goroutine并发打印的项目实践的文章就介绍到这了,更多相关Go Goroutine并发打印内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家! 

相关文章

  • go基于Gin框架的HTTP接口限速实践

    go基于Gin框架的HTTP接口限速实践

    HTTP接口在各个业务模块之间扮演着重要的角色,本文主要介绍了go基于Gin框架的HTTP接口限速实践,具有一定的参考价值,感兴趣的可以了解一下
    2023-09-09
  • go语言制作一个gif动态图

    go语言制作一个gif动态图

    这篇文章主要介绍了go制作一个gif动态图的相关资料,需要的朋友可以参考下
    2015-03-03
  • Go 并发编程协程及调度机制详情

    Go 并发编程协程及调度机制详情

    这篇文章主要介绍了Go并发编程协程及调度机制详情,协程是Go语言最大的特色之一,goroutine的实现其实是通过协程,更多相关内容需要的朋友可以参考一下
    2022-09-09
  • Golang中panic的异常处理

    Golang中panic的异常处理

    本文主要介绍了Golang中panic的异常处理,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2022-08-08
  • Go语言实现JSON解析的方法详解

    Go语言实现JSON解析的方法详解

    在日常项目中,使用Json格式进行数据封装是比较常见的操作。本文将详细讲解如何利用Go语言实现JSON的解析,感兴趣的小伙伴可以学习一下
    2022-04-04
  • 浅谈Golang是如何读取文件内容的(7种)

    浅谈Golang是如何读取文件内容的(7种)

    这篇文章主要介绍了浅谈Golang是如何读取文件内容的,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-05-05
  • GoLang string类型深入分析

    GoLang string类型深入分析

    string 作为 go 语言中的基础类型,其实有一些需要反复揣摩的,可能是我们使用的场景太简单,也可能是我们不需要那可怜的一点优化来提高性能,对它也就没那么上心了
    2023-01-01
  • go内置函数copy()的具体使用

    go内置函数copy()的具体使用

    当我们在Go语言中需要将一个切片的内容复制到另一个切片时,可以使用内置的copy()函数,本文就介绍了go内置函数copy()的具体使用,感兴趣的可以了解一下
    2023-08-08
  • 阿里云go开发环境搭建过程

    阿里云go开发环境搭建过程

    这篇文章主要介绍了阿里云go开发环境搭建过程,非常不错,具有参考借鉴价值,需要的朋友可以参考下
    2018-02-02
  • go run main.go 一直提示找不到包的解决方案

    go run main.go 一直提示找不到包的解决方案

    这篇文章主要介绍了go run main.go 一直提示找不到包的解决方案,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2021-04-04

最新评论