0

0

Go Goroutine 调度深度解析:理解并发与并行执行的差异

DDD

DDD

发布时间:2025-11-29 16:02:13

|

352人浏览过

|

来源于php中文网

原创

Go Goroutine 调度深度解析:理解并发与并行执行的差异

本文深入探讨 go 语言中 goroutine 的调度机制,解释为何在多 goroutine 场景下,实际的并行执行行为可能与预期不符。我们将阐明 go 运行时调度器的工作原理,特别是 gomaxprocs 参数对并发与并行执行的影响,并提供实践建议以实现更有效的并行处理。

1. Goroutine 与 Go 并发模型概述

Go 语言通过 Goroutine 提供了一种轻量级的并发机制。Goroutine 是由 Go 运行时管理的协程,相比传统的操作系统线程,它具有更小的内存开销和更快的创建、切换速度。当使用 go 关键字启动一个函数时,它就会在一个新的 Goroutine 中异步执行。

理解 Go 的并发模型,首先要区分“并发”(Concurrency)和“并行”(Parallelism):

  • 并发:指能够同时处理多个任务的能力,即使这些任务并非在同一时刻执行。它们可能在单个处理器核心上通过时间片轮转的方式交替执行。
  • 并行:指多个任务在多个处理器核心上真正地同时、同步执行。

Go 语言天生支持并发,但要实现真正的并行则需要满足一定的条件。

2. Go 运行时调度器的工作原理

Go 运行时包含一个复杂的调度器,负责将 Goroutine 映射到操作系统线程上执行。这个调度器采用了 M:N 调度模型,即 N 个 Goroutine 可以在 M 个操作系统线程上运行。其核心组件包括:

  • G (Goroutine):Go 应用程序中的并发执行单元。
  • M (Machine/OS Thread):操作系统线程,是实际执行代码的载体。
  • P (Processor):逻辑处理器,代表一个 Go 运行时上下文,它将 G 绑定到 M 上。每个 P 维护一个本地的 Goroutine 队列,当 M 完成当前 G 的执行后,会从 P 的队列中取出下一个 G 执行。当 P 的本地队列为空时,它会尝试从其他 P 的队列中“偷取” Goroutine。

2.1 GOMAXPROCS 的作用

GOMAXPROCS 是一个环境变量,也可以通过 runtime.GOMAXPROCS 函数在程序中设置。它控制了 Go 运行时可以同时使用的逻辑处理器 P 的数量。

PixVerse
PixVerse

PixVerse是一款强大的AI视频生成工具,可以轻松地将多种输入转化为令人惊叹的视频。

下载
  • 默认值:在 Go 1.5 版本及之后,GOMAXPROCS 的默认值通常设置为机器的 CPU 核心数 (runtime.NumCPU())。但在更早的版本中或某些特定环境下,其默认值可能为 1。
  • GOMAXPROCS = 1 的影响:当 GOMAXPROCS 设置为 1 时,即使系统拥有多个 CPU 核心,Go 运行时也只会创建一个逻辑处理器 P。这意味着所有 Goroutine 都必须在一个操作系统线程(M)上轮流执行。在这种情况下,Goroutine 之间只能实现并发,而无法实现真正的并行执行。它们会分时共享 CPU 资源,表现为交替执行。
  • GOMAXPROCS > 1 的影响:当 GOMAXPROCS 设置为大于 1 的值时,Go 运行时会创建相应数量的逻辑处理器 P。这些 P 可以被多个操作系统线程 M 使用,从而允许 Goroutine 在多个 CPU 核心上同时执行,实现并行。

3. 实验现象分析:为何并行执行不如预期?

在提供的实验代码中,用户启动了两个计算密集型(CPU 密集型)的 Goroutine,每个 Goroutine 都重复调用 fib 函数。观察到的现象是,在大部分时间内,两个 Goroutine 都是以大块的形式交替执行,而不是预期的“并排”或细粒度交错执行,直到任务后期才出现更频繁的切换。

这种现象通常是 GOMAXPROCS 默认值为 1 时的一个典型表现。当 GOMAXPROCS=1 时:

  1. 单线程执行:Go 运行时只有一个逻辑处理器 P,所有 Goroutine 都在这一个 P 绑定的 M(操作系统线程)上执行。
  2. 长时间占用 CPU:fib 函数是一个纯粹的 CPU 密集型计算,它不会主动进行 I/O 操作,也不会调用 runtime.Gosched() 等函数来主动让出 CPU。这意味着一个 Goroutine 一旦开始执行 fib 计算,它会尽可能长时间地占用 CPU,直到被 Go 调度器强制抢占(preemption)。
  3. 抢占式调度:Go 调度器在 Goroutine 运行时会定期进行抢占检查。在 Go 1.14 之前,抢占主要发生在函数调用或循环的后向跳转点。从 Go 1.14 开始,Go 引入了非协作式抢占(asynchronous preemption),使得调度器可以在任意安全点抢占 Goroutine。然而,即使有了抢占,对于长时间运行的 CPU 密集型任务,调度器也可能允许一个 Goroutine 运行相当长一段时间,直到达到抢占点或时间片耗尽。
  4. “后期交错”现象:至于为何在任务后期出现更频繁的交错,这可能与 fib 函数在计算较大数字时所需的执行时间有关,或者仅仅是调度器在特定时刻的决策。调度器的行为是非确定性的,它会根据内部启发式算法和系统负载来决定何时切换 Goroutine。在 CPU 密集型任务中,如果一个 Goroutine 不主动让出 CPU,调度器倾向于让它运行一段时间,以减少上下文切换的开销。

4. 实现真正的并行执行

要让 Goroutine 在多核 CPU 上实现真正的并行执行,关键在于确保 Go 运行时能够利用多个逻辑处理器 P。最直接的方法是设置 GOMAXPROCS。

4.1 设置 GOMAXPROCS

你可以通过以下两种方式设置 GOMAXPROCS:

  • 通过代码设置 (推荐): 在 main 函数的开头调用 runtime.GOMAXPROCS 函数。通常,我们会将其设置为机器的 CPU 核心数,以充分利用硬件资源。

    package main
    
    import (
        "fmt"
        "runtime"
        "sync" // 用于等待 Goroutine 完成
    )
    
    // fib 计算第 n 个斐波那

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
线程和进程的区别
线程和进程的区别

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

766

2023.08.10

Java 并发编程高级实践
Java 并发编程高级实践

本专题深入讲解 Java 在高并发开发中的核心技术,涵盖线程模型、Thread 与 Runnable、Lock 与 synchronized、原子类、并发容器、线程池(Executor 框架)、阻塞队列、并发工具类(CountDownLatch、Semaphore)、以及高并发系统设计中的关键策略。通过实战案例帮助学习者全面掌握构建高性能并发应用的工程能力。

100

2025.12.01

页面置换算法
页面置换算法

页面置换算法是操作系统中用来决定在内存中哪些页面应该被换出以便为新的页面提供空间的算法。本专题为大家提供页面置换算法的相关文章,大家可以免费体验。

500

2023.08.14

TypeScript类型系统进阶与大型前端项目实践
TypeScript类型系统进阶与大型前端项目实践

本专题围绕 TypeScript 在大型前端项目中的应用展开,深入讲解类型系统设计与工程化开发方法。内容包括泛型与高级类型、类型推断机制、声明文件编写、模块化结构设计以及代码规范管理。通过真实项目案例分析,帮助开发者构建类型安全、结构清晰、易维护的前端工程体系,提高团队协作效率与代码质量。

48

2026.03.13

Python异步编程与Asyncio高并发应用实践
Python异步编程与Asyncio高并发应用实践

本专题围绕 Python 异步编程模型展开,深入讲解 Asyncio 框架的核心原理与应用实践。内容包括事件循环机制、协程任务调度、异步 IO 处理以及并发任务管理策略。通过构建高并发网络请求与异步数据处理案例,帮助开发者掌握 Python 在高并发场景中的高效开发方法,并提升系统资源利用率与整体运行性能。

88

2026.03.12

C# ASP.NET Core微服务架构与API网关实践
C# ASP.NET Core微服务架构与API网关实践

本专题围绕 C# 在现代后端架构中的微服务实践展开,系统讲解基于 ASP.NET Core 构建可扩展服务体系的核心方法。内容涵盖服务拆分策略、RESTful API 设计、服务间通信、API 网关统一入口管理以及服务治理机制。通过真实项目案例,帮助开发者掌握构建高可用微服务系统的关键技术,提高系统的可扩展性与维护效率。

270

2026.03.11

Go高并发任务调度与Goroutine池化实践
Go高并发任务调度与Goroutine池化实践

本专题围绕 Go 语言在高并发任务处理场景中的实践展开,系统讲解 Goroutine 调度模型、Channel 通信机制以及并发控制策略。内容包括任务队列设计、Goroutine 池化管理、资源限制控制以及并发任务的性能优化方法。通过实际案例演示,帮助开发者构建稳定高效的 Go 并发任务处理系统,提高系统在高负载环境下的处理能力与稳定性。

59

2026.03.10

Kotlin Android模块化架构与组件化开发实践
Kotlin Android模块化架构与组件化开发实践

本专题围绕 Kotlin 在 Android 应用开发中的架构实践展开,重点讲解模块化设计与组件化开发的实现思路。内容包括项目模块拆分策略、公共组件封装、依赖管理优化、路由通信机制以及大型项目的工程化管理方法。通过真实项目案例分析,帮助开发者构建结构清晰、易扩展且维护成本低的 Android 应用架构体系,提升团队协作效率与项目迭代速度。

99

2026.03.09

JavaScript浏览器渲染机制与前端性能优化实践
JavaScript浏览器渲染机制与前端性能优化实践

本专题围绕 JavaScript 在浏览器中的执行与渲染机制展开,系统讲解 DOM 构建、CSSOM 解析、重排与重绘原理,以及关键渲染路径优化方法。内容涵盖事件循环机制、异步任务调度、资源加载优化、代码拆分与懒加载等性能优化策略。通过真实前端项目案例,帮助开发者理解浏览器底层工作原理,并掌握提升网页加载速度与交互体验的实用技巧。

105

2026.03.06

热门下载

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

精品课程

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

共32课时 | 6.2万人学习

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

共10课时 | 0.9万人学习

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

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