0

0

如何在Golang中实现并发排序_Golang goroutine与channel排序示例

P粉602998670

P粉602998670

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

|

777人浏览过

|

来源于php中文网

原创

Go需手动实现并发排序:按CPU核数切分数据→各goroutine独立排序子数组→channel传递结果→主goroutine多路归并;关键在避免底层数组重叠、合理设阈值、归并保序。

如何在golang中实现并发排序_golang goroutine与channel排序示例

goroutine 分割数据再合并排序结果

Go 本身不提供内置的并发排序函数,sort.Sort 是单线程的。若想利用多核加速排序,需手动切分数据、并发排序子数组、再归并。关键不是“开一堆 goroutine 排同一个 slice”,而是合理划分 + 安全归并。

常见错误是直接对共享 []int 启动多个 sort.Ints —— 这不仅不加速,还可能因数据重叠导致结果错乱。

  • 按 CPU 核心数或预设阈值(如每段 ≥ 10000 元素)切分原 slice
  • 每个 goroutine 处理一个独立子 slice,调用 sort.Ints 或自定义 sort.Sort
  • 用 channel 收集各段已排序结果,主 goroutine 负责多路归并(类似 merge sort 的 merge 步骤)

用 channel 实现安全的并发归并

归并阶段不能简单把所有子结果 append 到一起——必须保持全局有序。典型做法是启动一个归并 goroutine,从多个排序完成的 channel 中取最小值。

为避免阻塞和资源泄漏,建议用带缓冲的 channel 传递子结果(例如 chan []int),并在发送前确保子 slice 已完全排序完毕。

立即学习go语言免费学习笔记(深入)”;

  • 每个排序 goroutine 完成后,将结果写入同一 chan []int
  • 主 goroutine 用 for range 接收所有子结果,存入切片切片 [][]int
  • 调用自写 mergeSortedSlices 函数归并(非并发,但时间复杂度可控)

注意 sort.Slice 的并发安全性

sort.Slice 本身无并发问题,但它操作的是传入的 slice 底层数组。如果多个 goroutine 同时对**同一底层数组的不同 slice 视图**调用 sort.Slice,只要这些视图互不重叠,就是安全的。

HTTPie AI
HTTPie AI

AI API开发工具

下载

容易踩的坑:

  • 误用 data[i:j]data[k:l] 时,底层指针相同且区间重叠 → 数据被多次改写
  • 未用 copy 隔离子 slice,导致后续读取看到中间态数据
  • 忘记在 goroutine 中 recover panic(比如传入 nil slice 给 sort.Slice

一个可运行的 minimal 示例

以下代码实现:将 []int 均分为 4 段,并发排序,再归并。适用于中等规模数据(几万到百万级),超过千万建议改用外部排序或更精细的分治策略。

package main

import ( "sort" "sync" )

func concurrentMergeSort(data []int, numWorkers int) []int { n := len(data) if n <= 1 { return data } if numWorkers < 1 { numWorkers = 1 }

ch := make(chan []int, numWorkers)
var wg sync.WaitGroup

// 切分并启动 goroutine
segmentSize := n / numWorkers
for i := 0; i < numWorkers; i++ {
    start := i * segmentSize
    end := start + segmentSize
    if i == numWorkers-1 {
        end = n // 最后一段收尾
    }
    if start >= n {
        break
    }
    wg.Add(1)
    go func(s, e int) {
        defer wg.Done()
        segment := make([]int, e-s)
        copy(segment, data[s:e])
        sort.Ints(segment)
        ch <- segment
    }(start, end)
}
go func() {
    wg.Wait()
    close(ch)
}()

// 收集并归并
var sortedSegments [][]int
for seg := range ch {
    sortedSegments = append(sortedSegments, seg)
}
return mergeSortedSlices(sortedSegments)

}

func mergeSortedSlices(segments [][]int) []int { if len(segments) == 0 { return nil } if len(segments) == 1 { return segments[0] }

result := make([]int, 0, 1024)
indices := make([]int, len(segments))

for {
    minVal := 0
    minIdx := -1
    for i, seg := range segments {
        if indices[i] >= len(seg) {
            continue
        }
        if minIdx == -1 || seg[indices[i]] < minVal {
            minVal = seg[indices[i]]
            minIdx = i
        }
    }
    if minIdx == -1 {
        break
    }
    result = append(result, minVal)
    indices[minIdx]++
}
return result

}

真正难的不是开 goroutine,而是切分边界计算、归并时的内存局部性、以及小数据量下并发反而拖慢(调度开销 > 计算收益)。实测时建议用 benchstat 对比 sort.Ints 和你的并发版本在不同长度下的表现。

相关专题

更多
golang如何定义变量
golang如何定义变量

golang定义变量的方法:1、声明变量并赋予初始值“var age int =值”;2、声明变量但不赋初始值“var age int”;3、使用短变量声明“age :=值”等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

174

2024.02.23

golang有哪些数据转换方法
golang有哪些数据转换方法

golang数据转换方法:1、类型转换操作符;2、类型断言;3、字符串和数字之间的转换;4、JSON序列化和反序列化;5、使用标准库进行数据转换;6、使用第三方库进行数据转换;7、自定义数据转换函数。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

224

2024.02.23

golang常用库有哪些
golang常用库有哪些

golang常用库有:1、标准库;2、字符串处理库;3、网络库;4、加密库;5、压缩库;6、xml和json解析库;7、日期和时间库;8、数据库操作库;9、文件操作库;10、图像处理库。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

335

2024.02.23

golang和python的区别是什么
golang和python的区别是什么

golang和python的区别是:1、golang是一种编译型语言,而python是一种解释型语言;2、golang天生支持并发编程,而python对并发与并行的支持相对较弱等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

206

2024.03.05

golang是免费的吗
golang是免费的吗

golang是免费的。golang是google开发的一种静态强类型、编译型、并发型,并具有垃圾回收功能的开源编程语言,采用bsd开源协议。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

388

2024.05.21

golang结构体相关大全
golang结构体相关大全

本专题整合了golang结构体相关大全,想了解更多内容,请阅读专题下面的文章。

193

2025.06.09

golang相关判断方法
golang相关判断方法

本专题整合了golang相关判断方法,想了解更详细的相关内容,请阅读下面的文章。

188

2025.06.10

golang数组使用方法
golang数组使用方法

本专题整合了golang数组用法,想了解更多的相关内容,请阅读专题下面的文章。

191

2025.06.17

小游戏4399大全
小游戏4399大全

4399小游戏免费秒玩大全来了!无需下载、即点即玩,涵盖动作、冒险、益智、射击、体育、双人等全品类热门小游戏。经典如《黄金矿工》《森林冰火人》《狂扁小朋友》一应俱全,每日更新最新H5游戏,支持电脑与手机跨端畅玩。访问4399小游戏中心,重温童年回忆,畅享轻松娱乐时光!官方入口安全绿色,无插件、无广告干扰,打开即玩,快乐秒达!

30

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号