0

0

Go 并发编程:解决通道死锁问题

聖光之護

聖光之護

发布时间:2025-10-03 16:45:05

|

523人浏览过

|

来源于php中文网

原创

go 并发编程:解决通道死锁问题

本文旨在帮助开发者理解并解决 Go 语言并发编程中常见的通道死锁问题。通过分析一个简单的求和示例,我们将深入探讨死锁产生的原因,并提供两种有效的解决方案:利用计数器替代 range 循环,避免对未关闭通道的无限等待。掌握这些技巧,可以有效提升 Go 并发程序的健壮性和可靠性。

在 Go 语言的并发编程中,通道(channel)扮演着至关重要的角色,用于 goroutine 之间的通信和同步。然而,不当的使用方式容易导致死锁,影响程序的正常运行。本文将通过一个实际的例子,深入剖析死锁的产生原因,并提供解决方案。

死锁示例:并发求和

考虑以下场景:我们需要将一个整数数组分割成两部分,然后使用两个 goroutine 分别计算各自部分的和,最后将两个结果汇总并输出。

package main

import (
    "fmt"
)

// Add calculates the sum of elements in a and sends the result to res.
func Add(a []int, res chan<- int) {
    sum := 0
    for _, v := range a {
        sum += v
    }
    res <- sum
}

func main() {
    a := []int{1, 2, 3, 4, 5, 6, 7}

    n := len(a)
    ch := make(chan int)
    go Add(a[:n/2], ch)
    go Add(a[n/2:], ch)

    sum := 0
    for s := range ch {
        sum += s
    }

    fmt.Println(sum)
}

这段代码存在死锁的风险。原因在于 main 函数中的 for s := range ch 循环会持续尝试从通道 ch 中接收数据,直到通道关闭。然而,Add 函数在发送完各自的和之后并没有关闭通道,导致 range 循环永远无法结束,从而产生死锁。

解决方案一:使用计数器

一种解决方案是使用计数器来控制循环的结束。我们可以预先知道将会有多少个 goroutine 向通道发送数据,然后在主 goroutine 中使用一个计数器来记录已接收到的数据数量。当计数器达到预期值时,循环结束。

package main

import (
    "fmt"
)

// Add calculates the sum of elements in a and sends the result to res.
func Add(a []int, res chan<- int) {
    sum := 0
    for _, v := range a {
        sum += v
    }
    res <- sum
}

func main() {
    a := []int{1, 2, 3, 4, 5, 6, 7}

    n := len(a)
    ch := make(chan int)
    go Add(a[:n/2], ch)
    go Add(a[n/2:], ch)

    sum := 0
    count := 0 // Initialize the counter

    for count < 2 { // Loop until all results are received
        s := <-ch
        sum += s
        count++ // Increment the counter
    }

    fmt.Println(sum)
}

在这个版本中,我们添加了一个 count 变量来跟踪从通道接收到的结果数量。循环只会在 count 小于 2 时继续,确保了在接收到两个结果后循环能够正常结束,避免了死锁。

Groq
Groq

GroqChat是一个全新的AI聊天机器人平台,支持多种大模型语言,可以免费在线使用。

下载

解决方案二:关闭通道

另一种解决方案是在所有发送者完成发送后关闭通道。这样,range 循环就能检测到通道已关闭,并正常结束。但是,在并发环境中,确定所有发送者都已完成发送可能比较困难。

以下代码展示了如何在 Add 函数完成后关闭通道(不推荐,仅作演示):

package main

import (
    "fmt"
    "sync"
)

// Add calculates the sum of elements in a and sends the result to res.
func Add(a []int, res chan<- int, wg *sync.WaitGroup) {
    defer wg.Done()
    sum := 0
    for _, v := range a {
        sum += v
    }
    res <- sum
}

func main() {
    a := []int{1, 2, 3, 4, 5, 6, 7}

    n := len(a)
    ch := make(chan int)
    var wg sync.WaitGroup

    wg.Add(2)
    go Add(a[:n/2], ch, &wg)
    go Add(a[n/2:], ch, &wg)

    go func() {
        wg.Wait()
        close(ch)
    }()

    sum := 0
    for s := range ch {
        sum += s
    }

    fmt.Println(sum)
}

注意: 在多个 goroutine 向同一个通道发送数据时,直接在发送者 goroutine 中关闭通道通常是不安全的。上面的示例使用 sync.WaitGroup 确保所有 Add 函数完成后再关闭通道,但这种方法相对复杂,并且容易出错。因此,通常推荐使用计数器的方式来避免死锁。

总结

Go 语言的通道是强大的并发工具,但使用不当容易导致死锁。理解死锁的产生原因,并掌握合适的解决方案至关重要。本文介绍了两种常用的解决方案:使用计数器和关闭通道。在实际开发中,应根据具体情况选择最合适的方案,确保程序的健壮性和可靠性。通常,使用计数器是更安全和推荐的做法。

相关文章

编程速学教程(入门课程)
编程速学教程(入门课程)

编程怎么学习?编程怎么入门?编程在哪学?编程怎么学才快?不用担心,这里为大家提供了编程速学教程(入门课程),有需要的小伙伴保存下载就能学习啦!

下载

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

相关专题

更多
counta和count的区别
counta和count的区别

Count函数用于计算指定范围内数字的个数,而CountA函数用于计算指定范围内非空单元格的个数。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

193

2023.11.20

Golang channel原理
Golang channel原理

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

241

2025.11.14

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

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

320

2025.11.17

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

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

65

2025.12.31

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

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

45

2025.12.31

视频文件格式
视频文件格式

本专题整合了视频文件格式相关内容,阅读专题下面的文章了解更多详细内容。

40

2025.12.31

不受国内限制的浏览器大全
不受国内限制的浏览器大全

想找真正自由、无限制的上网体验?本合集精选2025年最开放、隐私强、访问无阻的浏览器App,涵盖Tor、Brave、Via、X浏览器、Mullvad等高自由度工具。支持自定义搜索引擎、广告拦截、隐身模式及全球网站无障碍访问,部分更具备防追踪、去谷歌化、双内核切换等高级功能。无论日常浏览、隐私保护还是突破地域限制,总有一款适合你!

41

2025.12.31

出现404解决方法大全
出现404解决方法大全

本专题整合了404错误解决方法大全,阅读专题下面的文章了解更多详细内容。

232

2025.12.31

html5怎么播放视频
html5怎么播放视频

想让网页流畅播放视频?本合集详解HTML5视频播放核心方法!涵盖<video>标签基础用法、多格式兼容(MP4/WebM/OGV)、自定义播放控件、响应式适配及常见浏览器兼容问题解决方案。无需插件,纯前端实现高清视频嵌入,助你快速打造现代化网页视频体验。

9

2025.12.31

热门下载

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

精品课程

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

共32课时 | 3.2万人学习

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号