0

0

Go Channel 与 Goroutine 中指针变量引发的死锁问题解析

聖光之護

聖光之護

发布时间:2025-12-31 23:27:09

|

890人浏览过

|

来源于php中文网

原创

Go Channel 与 Goroutine 中指针变量引发的死锁问题解析

本文详解 go 中因未关闭 channel 和错误使用 sync.waitgroup 导致“all goroutines are asleep - deadlock”错误的根本原因,并提供可复用的并发控制模板。

在 Go 并发编程中,all goroutines are asleep - deadlock 是一个典型且易被忽视的运行时错误。它并非语法或逻辑错误,而是程序陷入永久阻塞状态:所有 goroutine 都在等待(如从 channel 读取、写入或等待 WaitGroup),却无人唤醒或终止,导致主 goroutine 无法继续执行而崩溃。

在你提供的代码中,问题根源有两点:

  1. sync.WaitGroup 计数不匹配:wg.Add(len(urls)) 错误地为每个 URL 增加计数,但实际只启动了 1 个 Poller goroutine(而非 len(urls) 个)。WaitGroup 的 Add() 应与 Done() 调用次数严格对应——即每个 goroutine 生命周期 对应一次 Add(1) + Done()。当前代码中,Poller 函数内仅调用一次 wg.Done(),但 main 中却 Add(len(urls)),造成计数严重失衡,wg.Wait() 永远不会返回。

  2. channel 未关闭导致无限阻塞:Poller 使用 for r := range in 循环持续接收数据。该循环仅在 channel 被显式关闭后才会退出。而你的写入 goroutine 在发送完所有 URL 后直接结束,既未关闭 pending channel,也未调用 wg.Done() 标记自身完成,导致 Poller 永远卡在 range 等待下一条数据,主 goroutine 则在 wg.Wait() 处死锁。

✅ 正确做法是:

Batch GPT
Batch GPT

使用AI批量处理数据、自动执行任务

下载
  • wg.Add(n) 中的 n 表示需等待的 goroutine 数量(此处为 2:1 个 Poller + 1 个发送器);
  • 发送器 goroutine 完成后,先关闭 channel,再调用 wg.Done()(顺序很重要:关闭必须在所有发送操作之后,且 Done() 标识自身退出);
  • Poller 中使用 defer wg.Done() 确保 goroutine 结束时正确减计数。

以下是修复后的完整可运行示例(已适配 numPollers = 2 的并发模型):

package main

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

const numPollers = 2 // 启动 2 个并发 Poller

var urls = []string{
    "http://www.google.com/",
    "http://golang.org/",
    "http://blog.golang.org/",
    "http://golangtutorials.blogspot.fr",
    "https://gobyexample.com/",
}

type Resource struct {
    url string
}

// Poller 从 channel 拉取 *Resource 并处理(此处仅打印)
func Poller(in <-chan *Resource, wg *sync.WaitGroup) {
    defer wg.Done()
    for r := range in { // range 会自动在 channel 关闭后退出
        fmt.Printf("Processed: %s - %s\n", r.url, time.Now().Format("15:04:05"))
    }
}

func main() {
    var wg sync.WaitGroup
    pending := make(chan *Resource, len(urls)) // 可选:加缓冲避免发送阻塞

    // 启动 numPollers 个 Poller goroutine
    wg.Add(numPollers)
    for i := 0; i < numPollers; i++ {
        go Poller(pending, &wg)
    }

    // 启动 1 个发送 goroutine:写入 URL 并关闭 channel
    wg.Add(1)
    go func() {
        defer close(pending) // ✅ 关键:发送完成后关闭 channel
        defer wg.Done()      // ✅ 标记发送器 goroutine 完成
        for _, url := range urls {
            fmt.Printf("Sending: %s\n", url)
            pending <- &Resource{url: url}
        }
    }()

    wg.Wait() // 等待所有 Poller 和发送器完成
    fmt.Printf("✅ All done at %s\n", time.Now().Format("15:04:05"))
}

? 关键注意事项

  • close(pending) 必须由唯一写入者(即发送 goroutine)调用,且只能调用一次;多个 goroutine 写入时需额外协调。
  • 若需限制并发请求数(如 HTTP 调用),应在 Poller 内部实现(例如用 http.Client 发起请求),而非依赖 channel 缓冲区大小。
  • 使用 make(chan *Resource, N) 设置缓冲区可避免发送端阻塞,但不解决死锁本质——channel 关闭仍是 range 退出的必要条件
  • defer 语句按后进先出(LIFO)执行,因此 close(pending) 会在 wg.Done() 之前执行,确保 Poller 能收到关闭信号。

掌握 channel 生命周期与 WaitGroup 计数的精确匹配,是写出健壮 Go 并发程序的基石。

相关专题

更多
resource是什么文件
resource是什么文件

Resource文件是一种特殊类型的文件,它通常用于存储应用程序或操作系统中的各种资源信息。它们在应用程序开发中起着关键作用,并在跨平台开发和国际化方面提供支持。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

141

2023.12.20

Golang channel原理
Golang channel原理

本专题整合了Golang channel通信相关介绍,阅读专题下面的文章了解更多详细内容。

239

2025.11.14

golang channel相关教程
golang channel相关教程

本专题整合了golang处理channel相关教程,阅读专题下面的文章了解更多详细内容。

320

2025.11.17

http500解决方法
http500解决方法

http500解决方法有检查服务器日志、检查代码错误、检查服务器配置、检查文件和目录权限、检查资源不足、更新软件版本、重启服务器或寻求专业帮助等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

267

2023.11.09

http请求415错误怎么解决
http请求415错误怎么解决

解决方法:1、检查请求头中的Content-Type;2、检查请求体中的数据格式;3、使用适当的编码格式;4、使用适当的请求方法;5、检查服务器端的支持情况。更多http请求415错误怎么解决的相关内容,可以阅读下面的文章。

386

2023.11.14

HTTP 503错误解决方法
HTTP 503错误解决方法

HTTP 503错误表示服务器暂时无法处理请求。想了解更多http错误代码的相关内容,可以阅读本专题下面的文章。

1051

2024.03.12

http与https有哪些区别
http与https有哪些区别

http与https的区别:1、协议安全性;2、连接方式;3、证书管理;4、连接状态;5、端口号;6、资源消耗;7、兼容性。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

1677

2024.08.16

php源码安装教程大全
php源码安装教程大全

本专题整合了php源码安装教程,阅读专题下面的文章了解更多详细内容。

3

2025.12.31

php网站源码教程大全
php网站源码教程大全

本专题整合了php网站源码相关教程,阅读专题下面的文章了解更多详细内容。

1

2025.12.31

热门下载

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

精品课程

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

共32课时 | 3.1万人学习

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

共10课时 | 0.8万人学习

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

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