Go Goroutine并发:理解与启用真正的并行处理

Go Goroutine并发:理解与启用真正的并行处理

本文深入探讨Go语言中goroutine的并发执行机制,特别是当goroutine数量多于默认处理器核心数时,如何通过runtime.GOMAXPROCS确保任务在多核CPU上实现真正的并行处理。文章通过冒泡排序示例,解释了goroutine看似同步完成的现象,并指导如何配置运行时参数以优化并行性能,实现预期的独立任务加速。

1. Go Goroutine并发执行的挑战

go语言中,goroutine是轻量级的并发执行单元。开发者通常期望启动多个goroutine后,它们能够独立并行运行,尤其是当任务负载不同时,轻量级任务应更快完成。然而,实际观察到的行为有时并非如此,例如,多个goroutine在处理不同大小的数据集时,其“完成”消息可能几乎同时出现,这让人误以为它们在相互等待。

考虑以下冒泡排序的例子,其中启动了三个goroutine,分别对不同大小的切片进行排序:

package mainimport (    "fmt"    "math/rand"    "time")/* 简单的冒泡排序算法 */func bubblesort(str string, a []int) []int {    for n := len(a); n > 1; n-- {        for i := 0; i  a[i+1] {                a[i], a[i+1] = a[i+1], a[i] // 交换            }        }    }    fmt.Println(str + " done") // 完成消息    return a}/* 用伪随机数填充切片 */func random_fill(a []int) []int {    for i := 0; i < len(a); i++ {        a[i] = rand.Int()    }    return a}func main() {    rand.Seed(time.Now().UTC().UnixNano()) // 设置随机数种子    a1 := make([]int, 34589) // 创建切片    a2 := make([]int, 42)    // 创建切片    a3 := make([]int, 9999)  // 创建切片    a1 = random_fill(a1) // 填充切片    a2 = random_fill(a2) // 填充切片    a3 = random_fill(a3) // 填充切片    fmt.Println("Slices filled ...")    go bubblesort("Thread 1", a1) // 1. Goroutine 启动    go bubblesort("Thread 2", a2) // 2. Goroutine 启动    go bubblesort("Thread 3", a3) // 3. Goroutine 启动    fmt.Println("Main working ...")    time.Sleep(1 * time.Minute) // 等待1分钟以接收"done"消息}

在某些环境下运行上述代码,可能会得到如下输出:

Slices filled ...Main working ...Thread 1 doneThread 2 doneThread 3 done

尽管 a2 切片最小(42个元素),a3 次之(9999个元素),a1 最大(34589个元素),但“done”消息却几乎同时出现,或者顺序不确定,且不总是反映任务的实际完成时间。这并非goroutine在相互等待,而是Go运行时调度器在默认配置下,可能没有充分利用多核CPU的并行能力。

2. Go调度器与GOMAXPROCS

Go语言的并发模型是基于M:N调度器实现的,它将M个goroutine调度到N个操作系统线程上执行。默认情况下,Go运行时会尝试利用所有可用的CPU核心。然而,在Go 1.5版本之前,runtime.GOMAXPROCS 的默认值是1,这意味着Go程序在任何给定时刻最多只能有一个操作系统线程在执行Go代码,即使系统有多个CPU核心,goroutine也只能通过时间片轮转的方式并发执行,而非真正的并行。

要强制Go运行时在多个CPU核心上并行执行goroutine,需要显式地设置 runtime.GOMAXPROCS。这个函数用于设置可以同时执行Go代码的操作系统线程的最大数量。

3. 实现真正的并行:配置GOMAXPROCS

为了让Go程序充分利用多核CPU,实现goroutine的真正并行,可以在 main 函数的开头调用 runtime.GOMAXPROCS。

package mainimport (    "fmt"    "math/rand"    "runtime" // 导入 runtime 包    "time")/* 简单的冒泡排序算法 */func bubblesort(str string, a []int) []int {    for n := len(a); n > 1; n-- {        for i := 0; i  a[i+1] {                a[i], a[i+1] = a[i+1], a[i] // 交换            }        }    }    fmt.Println(str + " done") // 完成消息    return a}/* 用伪随机数填充切片 */func random_fill(a []int) []int {    for i := 0; i < len(a); i++ {        a[i] = rand.Int()    }    return a}func main() {    // 设置 Go 运行时可以使用的最大操作系统线程数    // 这里设置为2,表示最多两个OS线程可以同时执行Go代码    // 也可以设置为 runtime.NumCPU() 来使用所有可用的CPU核心    runtime.GOMAXPROCS(2)     rand.Seed(time.Now().UTC().UnixNano()) // 设置随机数种子    a1 := make([]int, 34589) // 创建切片    a2 := make([]int, 42)    // 创建切片    a3 := make([]int, 9999)  // 创建切片    a1 = random_fill(a1) // 填充切片    a2 = random_fill(a2) // 填充切片    a3 = random_fill(a3) // 填充切片    fmt.Println("Slices filled ...")    go bubblesort("Thread 1", a1) // 1. Goroutine 启动    go bubblesort("Thread 2", a2) // 2. Goroutine 启动    go bubblesort("Thread 3", a3) // 3. Goroutine 启动    fmt.Println("Main working ...")    time.Sleep(1 * time.Minute) // 等待1分钟以接收"done"消息}

修改后的代码,在执行时,由于 runtime.GOMAXPROCS(2) 的设置,Go调度器现在可以同时在两个操作系统线程上执行goroutine。这意味着,如果系统有至少两个核心,那么两个goroutine可以真正并行运行。预期输出将反映任务负载的差异:

用Apache Spark进行大数据处理 用Apache Spark进行大数据处理

本文档主要讲述的是用Apache Spark进行大数据处理——第一部分:入门介绍;Apache Spark是一个围绕速度、易用性和复杂分析构建的大数据处理框架。最初在2009年由加州大学伯克利分校的AMPLab开发,并于2010年成为Apache的开源项目之一。 在这个Apache Spark文章系列的第一部分中,我们将了解到什么是Spark,它与典型的MapReduce解决方案的比较以及它如何为大数据处理提供了一套完整的工具。希望本文档会给有需要的朋友带来帮助;感

用Apache Spark进行大数据处理 0 查看详情 用Apache Spark进行大数据处理

Slices filled ...Main working ...Thread 2 done  // 最小的切片最先完成Thread 3 done  // 中等大小的切片次之Thread 1 done  // 最大的切片最后完成

4. 注意事项与最佳实践

runtime.GOMAXPROCS 的默认值:

Go 1.5及更高版本中,runtime.GOMAXPROCS 的默认值已更改为 runtime.NumCPU(),即默认情况下Go程序会尝试使用所有可用的CPU核心进行并行处理。因此,对于新版本的Go,通常无需显式设置 runtime.GOMAXPROCS 就能获得并行优势。如果您的Go版本较老(低于1.5),或者您希望限制Go运行时使用的核心数量,那么显式调用 runtime.GOMAXPROCS 仍然是必要的。

选择 GOMAXPROCS 的值:

通常,将其设置为 runtime.NumCPU() 是一个好的实践,这样Go程序就能充分利用机器的所有物理核心。在某些特定场景下,例如,当程序同时执行大量I/O操作(I/O密集型)时,GOMAXPROCS 的值可能需要根据实际情况进行调整,甚至可以略大于 runtime.NumCPU(),以允许在等待I/O时调度器切换到其他goroutine。然而,对于CPU密集型任务,通常不建议将其设置得远大于核心数,因为过多的OS线程切换会引入额外的开销。

Goroutine的调度顺序:

即使设置了 runtime.GOMAXPROCS,Go调度器对goroutine的执行顺序仍然不提供任何保证。上述示例中,Thread 2 最先完成是因为其任务量最小,而不是因为调度器优先选择了它。如果需要控制goroutine的执行顺序或等待所有goroutine完成,应使用 sync.WaitGroup、channel 等并发原语,而不是依赖 time.Sleep 这种粗糙的等待方式。

time.Sleep 的副作用:

在 bubblesort 函数中添加 time.Sleep(1) 会强制调度器进行上下文切换,从而可能使小任务在等待时让出CPU给其他goroutine,导致看起来任务是并行完成的。但这会引入不必要的延迟,并不能真实反映算法的执行效率。

总结

理解Go语言的并发模型和调度器行为对于编写高性能的并发程序至关重要。通过正确配置 runtime.GOMAXPROCS(尤其是在Go 1.5之前的版本或需要特定控制的场景),我们可以确保Go程序能够充分利用多核CPU的并行能力,从而让goroutine在独立任务中真正实现并行加速,并获得预期的性能表现。同时,应结合 sync.WaitGroup 等工具,更优雅地管理goroutine的生命周期和同步。

以上就是Go Goroutine并发:理解与启用真正的并行处理的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1166317.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月3日 00:12:15
下一篇 2025年12月3日 00:12:46

相关推荐

  • PHP 函数如何与 Go 交互:提升跨语言性能

    php 函数可通过 syscall.syscall 函数与 go 交互,提升跨语言性能。步骤如下:在 php 中创建函数 callgofunction,接受函数名称和参数数组。在 go 中声明要导出的函数,例如 gofunction(a, b uint64) uint64。编译 go 代码并加载 s…

    2025年12月9日
    000
  • 如果 PHP 失宠,我会选择哪种后端语言?

    作为一名经验丰富的后端开发人员,php 在我的职业生涯中发挥了重要作用。然而,科技格局瞬息万变,我们必须时刻做好迎接新挑战的准备。那么,如果今天 php 突然消失了,我会选择哪种后端语言来取代它呢?这是我的坦率见解。 1. Golang首先,我毫无疑问会选择Golang(Go语言)。为什么?因为Go…

    2025年12月9日 好文分享
    100
  • PHP 函数如何与 Go 交互?

    PHP 函数如何与 Go 交互 PHP 和 Go 是两种截然不同的编程语言,具有不同的语法和特性。然而,在某些情况下,您可能需要在 PHP 应用程序和 Go 服务之间进行交互。 方法 1:使用 HTTP 请求 您可以使用标准 HTTP 请求在 PHP 和 Go 之间发送数据。 立即学习“PHP免费学…

    2025年12月9日
    000
  • PHP 函数如何与 Go 交互

    php 和 go 可通过结合使用实现优势互补,php 擅长处理 http 请求和数据库交互,而 go 具有高性能和并发性。通过 go 创建服务,并通过 php 发出请求,可以实现语言间的交互。实战中,php 应用程序可通过 curl 请求访问由 go 编写并部署的 api,扩展应用程序功能。 使用 …

    2025年12月9日
    000
  • php有哪些算法

    PHP 提供了丰富的算法,包括以下类型:排序算法:冒泡排序、快速排序、归并排序搜索算法:线性搜索、二分搜索、哈希搜索数据结构:哈希表、数组、链表字符串算法:正则表达式、字符串比较、字符串转换数学算法:随机数生成、三角函数、复数计算 PHP 中的算法 PHP 提供了一系列丰富的算法,可以满足各种应用场…

    2025年12月9日
    000
  • php都有哪些算法

    PHP 中提供的算法包括:排序、搜索、数学、字符串、数据结构、加密和图形。选择算法取决于问题和性能要求,需考虑数据规模、类型、复杂度和实现难度。 PHP中的算法 PHP 是一门强大的编程语言,提供了广泛的算法来解决各种问题。常见的 PHP 算法包括: 排序算法 冒泡排序选择排序快速排序归并排序桶排序…

    2025年12月9日
    000
  • 币安交易所(binance)新手如何进行合约交易操作及防爆仓指南

    币安合约交易需先熟悉界面,包括交易对、K线图、委托区和仓位信息,重点关注强平价格;执行交易时选择交易对、设置杠杆(新手建议低倍)、下单类型及数量,确认后提交;开仓后应设置止盈止损以控制风险;逐仓模式下可追加保证金降低强平风险;根据风险偏好在全仓与逐仓间切换保证金模式,全仓风险更高但资金利用率高。 币…

    2025年12月9日
    000
  • 币安binance交易所官网直链 Binance网页版安全登录链接

    币安binance 是当前全球交易量领先的数字资产交易平台之一,提供现货、合约、理财等多类型服务。本文将围绕 币安官网直链 与 网页版安全登录流程 展开,帮助你快速、安全进入 binance 官方页面完成账户操作。 币安Binance官网访问入口 要登录币安网页版,可通过浏览器输入官方域名进入官网首…

    2025年12月9日
    000
  • 币安交易所全球官网入口 Binance官方认证APP下载地址

    币安 binance 是全球领先的数字资产交易平台,支持现货交易、期货合约、理财产品等多样化服务。本文将为你介绍 币安全球官网入口 及其 官方 app 下载方式,帮助你安全访问并安装。 币安全球官网入口 建议通过官方域名访问币安官网:— 在这里你可以完成注册、登录、资产管理、充值提现等操作。 币安官…

    2025年12月9日
    000
  • 一文了解币圈:以太坊在哪里诞生?怎么购买?有什么作用?

    以太坊(ethereum)作为区块链技术的重要里程碑,不仅是一种数字资产,更是一个支持去中心化应用的全球性开源平台。本文旨在为初学者快速梳理以太坊的起源、获取方式及其核心应用场景,帮助您构建一个清晰的认知框架。 一、以太坊的诞生 1、以太坊最初由程序员 Vitalik Buterin 在2013年提…

    2025年12月9日
    000
  • 狗狗币是什么类型的币 一文了解狗狗币

    狗狗币(Dogecoin),常被昵称为“狗狗币”,最初作为一个网络玩笑诞生,但现已发展成为全球知名的数字资产之一。本文将为您详细解析狗狗币的起源、技术特点及其独特的社区文化,帮助您全面了解这个从互联网迷因(Meme)中走出的特殊加密货币。 一、源于玩笑的诞生 1、狗狗币于2013年由软件工程师比利·…

    2025年12月9日
    000
  • 欧易(OKX)交易所注册地址及APP下载地址

    OKX是全球数字资产服务平台,用户可通过官网网页端或移动端App注册。网页端注册需访问官方网址www.okx.com/join,填写邮箱或手机号、设置密码、完成人机验证并输入短信或邮件验证码;移动端则需通过手机浏览器下载对应系统的App,安装后打开应用,按提示完成注册流程。两种方式均需阅读并同意服务…

    2025年12月9日
    000
  • 币安为什么会是全球交易量最大的加密货币交易所?优势详解

    币安凭借高流动性、丰富产品、安全技术及全球化生态领先行业:其庞大用户基础和高效撮合系统保障交易深度与低滑点,多元币种与衍生品满足各类投资需求,高性能引擎与多重安全机制确保稳定与资产安全,全球布局与自建公链生态增强用户粘性,形成可持续发展的行业龙头优势。 币安Binance 币安Binance官网入口…

    2025年12月9日
    000
  • 一文读懂:狗狗币和小狗币的区别

    狗狗币(dogecoin)与小狗币(shiba inu)虽然都源于同一个网络迷因,并常被相提并论,但它们在技术基础、市场定位和发展目标上存在显著差异。本文将从多个维度深入解析,帮助您清晰地分辨这两种备受关注的数字资产。 一、出身与起源 1、狗狗币 (DOGE):诞生于2013年,由两位软件工程师作为…

    2025年12月9日
    000
  • 怎么几千块进场币圈快速翻百倍?

    1、%ignore_a_1%Binance 币安Binance官网入口: 币安BinanceAPP下载链接: 2、欧易okx 欧易okx官网入口: 欧易okxAPP下载链接: 3、火币HTX 官网入口: APP下载链接: 在数字资本资产市场中,利用有限的寻求高倍数增长是部分参与者的目标。这通常涉及高…

    2025年12月9日
    000
  • 新手小白怎么买比特币?怎么选交易平台?

    比特币作为一种创新的数字资产,近年来引起了全球范围内的广泛关注。对于许多刚接触数字货币领域的新手来说,如何安全、便捷地获取比特币,并选择一个可靠的交易平台,是他们迈入这个世界的第一步。 比特币的购买过程并非遥不可及,但却需要一定的知识储备和细致的考量。从理解数字资产的基本概念,到掌握交易平台的选择标…

    2025年12月9日
    000
  • 全球主流加密交易所盘点_2025年合规平台前十名推荐

    币安、OKX、火币、Coinbase、Kraken、Bybit、KuCoin、Bitstamp、Gemini和Bitfinex是全球主流加密交易平台。币安以高交易量和全球合规布局著称;OKX在衍生品领域突出并获迪拜与巴哈马监管批准;火币覆盖多国合规许可并推出数字资产消费卡。 选择一个具备合规资质且信…

    2025年12月9日
    000
  • 发明狗狗币的人有哪些?狗狗币详细介绍解析

    狗狗币(Dogecoin)作为加密货币领域的“幽默大师”,凭借其独特的社区文化和名人效应,早已从最初的玩笑演变成了市值巨大的主流资产。本文将深入揭秘其背后的创始团队,并全方位解析其技术特点与市场价值。 一、狗狗币的两位核心发明人 1、杰克逊·帕尔默 (Jackson Palmer):当时是Adobe…

    2025年12月9日
    000
  • 还会有下一个百倍币吗?2025年值得关注的五大新兴加密货币赛道

    1、币安Binance 币安Binance官网入口: 币安BinanceAPP下载链接: 2、欧易okx 欧易okx官网入口: 欧易okxAPP下载链接: 3、火币HTX 官网入口: APP下载链接: 在快速变化的加密市场中,识别增长的极限是投资者关注的焦点潜力。新兴的叙述和技术突破往往能催生出新的…

    2025年12月9日
    000
  • 一文读懂:莱特和狗狗币哪个值钱?在哪里能买到?

    莱特币(ltc)和狗狗币(doge)都是加密世界里的知名角色,但它们的价值逻辑和市场定位截然不同。本文将从价值、技术和社区文化等角度对比两者,并介绍获取它们的主流渠道。 一、价值对比:不能只看单价 1、从单个币的价格来看,莱特币(LTC)通常远高于狗狗币(DOGE)。但这并不能完全代表“谁更值钱”。…

    2025年12月9日
    000

发表回复

登录后才能评论
关注微信