0

0

Go 并行程序性能优化:深入分析与实践

聖光之護

聖光之護

发布时间:2025-08-28 19:33:00

|

840人浏览过

|

来源于php中文网

原创

go 并行程序性能优化:深入分析与实践

本文针对 Go 语言并行程序中出现的性能瓶颈问题,以一个大整数分解的例子入手,深入分析了 big.Int 类型在并行计算中的性能问题根源,并提供了优化建议。文章重点讨论了内存分配对并行性能的影响,并指出了程序中潜在的并发安全问题,旨在帮助读者更好地理解和优化 Go 并行程序。

性能瓶颈分析:big.Int 与内存分配

在 Go 语言中,big.Int 类型用于处理任意精度的整数。然而,在高并发场景下,频繁使用 big.Int 可能会导致性能问题。问题在于,big.Int 的许多方法(如 Mod、Add 等)需要在堆上分配内存来存储计算结果。由于 Go 语言的堆是全局共享的,所有 goroutine 都在竞争同一块内存区域,导致内存分配操作串行化,从而限制了并行程序的性能。

例如,在提供的代码中,factorize 函数中频繁使用 m.Mod(n, i) 进行取模运算。每次调用 Mod 都会分配新的 big.Int 对象来存储余数,这在高并发场景下会造成严重的性能瓶颈。

优化方案

  1. 避免不必要的 big.Int 使用: 如果可以,尽量使用内置的整数类型(如 int64)代替 big.Int。在示例代码中,如果被分解的数和可能的因子都在 int64 的范围内,则可以使用 int64 类型进行计算,从而避免内存分配开销。

    package main
    
    import (
        "fmt"
        "runtime"
        "sync"
    )
    
    func factorize(n int64, start int64, step int64, c chan int64) {
        for i := start; ; i += step {
            if n%i == 0 {
                c <- i
                return // 找到一个因子后退出
            }
        }
    }
    
    func main() {
        numCPUs := runtime.NumCPU()
        runtime.GOMAXPROCS(numCPUs)
    
        n := int64(28808539627864609)
        numProcesses := numCPUs // 使用 CPU 核心数作为进程数
        c := make(chan int64)
    
        var wg sync.WaitGroup
        wg.Add(numProcesses)
    
        for i := 0; i < numProcesses; i++ {
            go func(i int) {
                defer wg.Done()
                factorize(n, int64(2+i), int64(numProcesses), c)
            }(i)
        }
    
        go func() {
            wg.Wait()
            close(c) // 关闭 channel
        }()
    
        factor := <-c // 从 channel 接收结果
        fmt.Println("Factor:", factor)
    }

    注意事项: 上述代码使用了 sync.WaitGroup 来确保所有 goroutine 都完成后再关闭 channel,并使用了 return 语句在找到一个因子后退出 goroutine,避免了后续的并发安全问题。

  2. 减少内存分配: 如果必须使用 big.Int,尽量复用已有的 big.Int 对象,避免频繁创建新的对象。例如,可以将 m 声明在循环外部,然后在每次迭代中更新其值。

    func factorize(n *big.Int, start int, step int, c chan *big.Int) {
        var m big.Int
        i := big.NewInt(int64(start))
        s := big.NewInt(int64(step))
        z := big.NewInt(0)
    
        for {
            m.Mod(n, i) // 复用 m
            if m.Cmp(z) == 0{
                c <- big.NewInt(0).Set(i) // 发送 i 的拷贝
                return // 找到一个因子后退出
            }
            i.Add(i, s)
        }
    }

    注意事项: 由于 big.Int 是指针类型,直接发送 i 的指针会导致并发安全问题。因此,需要创建一个新的 big.Int 对象,并将 i 的值复制到新的对象中,再发送新的对象的指针。

    知我AI
    知我AI

    一款多端AI知识助理,通过一键生成播客/视频/文档/网页文章摘要、思维导图,提高个人知识获取效率;自动存储知识,通过与知识库聊天,提高知识利用效率。

    下载
  3. 使用更高效的算法: 在 big.Int 的方法中,不同的方法性能差异很大。例如,QuoRem 方法同时计算商和余数,通常比单独使用 Mod 和 Quo 方法更高效。

  4. 调整 GOMAXPROCS: runtime.GOMAXPROCS 设置了 Go 程序可以同时使用的 CPU 核心数。将其设置为 CPU 核心数可以提高并行程序的性能。

并发安全问题

原代码存在一个并发安全问题:当一个 goroutine 找到一个因子并通过 channel 发送结果时,它发送的是指向局部变量 i 的指针。但是,该 goroutine 并没有退出循环,而是继续递增 i 的值。当主 goroutine 从 channel 接收到指针时,i 的值可能已经被修改,导致结果错误。

为了解决这个问题,需要在找到一个因子后退出 goroutine。此外,由于 big.Int 是指针类型,发送指针可能会导致数据竞争。因此,应该发送 i 的拷贝,而不是 i 的指针。

总结

Go 语言的并发模型非常强大,但要充分利用其优势,需要深入理解其底层机制。在使用 big.Int 类型时,需要特别注意内存分配对性能的影响,并避免潜在的并发安全问题。通过合理的优化,可以显著提高 Go 并行程序的性能。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

WorkBuddy
WorkBuddy

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

1071

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

617

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

335

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

235

2025.08.29

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

448

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

606

2023.08.10

Golang channel原理
Golang channel原理

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

261

2025.11.14

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

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

352

2025.11.17

chatgpt使用指南
chatgpt使用指南

本专题整合了chatgpt使用教程、新手使用说明等等相关内容,阅读专题下面的文章了解更多详细内容。

0

2026.03.16

热门下载

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

精品课程

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

共32课时 | 6.3万人学习

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号