当前位置: 首页 > article >正文

Go-并行编程新手指南

Go 并行编程新手指南

在Go语言中,并行编程是充分利用多核CPU资源、提升程序性能的重要手段。它的核心概念包括goroutine和channel,这些特性使得Go在处理并发任务时表现出色。

goroutine:轻量级的并发执行单元

goroutine是Go并行编程的基础。它类似于线程,但更为轻量级。与传统线程相比,创建和销毁goroutine的开销极小,且栈空间可按需动态增长。在Go语言中,只需在函数调用前加上go关键字,就能轻松创建一个新的goroutine。例如:

go list.Sort()

这样,list.Sort()函数就会在一个新的goroutine中并发执行,而不会阻塞当前的执行流程。你还可以使用函数字面量来创建更灵活的goroutine,像这样:

func Announce(message string, delay time.Duration) {
    go func() {
        time.Sleep(delay)
        fmt.Println(message)
    }()
}

这种方式常用于需要延迟执行或异步处理的场景。

channel:实现数据共享与同步的桥梁

channel用于在不同的goroutine之间进行通信和同步。它就像是一个管道,goroutine可以通过它发送和接收数据。channel分为无缓冲通道和有缓冲通道。无缓冲通道在发送和接收数据时会进行同步操作,即发送者会阻塞,直到有接收者接收数据;而有缓冲通道则允许在缓冲区未满时,发送者无需等待接收者即可发送数据。创建channel的方式如下:

ci := make(chan int)            // 无缓冲的整数通道
cs := make(chan *os.File, 100)  // 有100个元素缓冲的文件指针通道

在实际应用中,channel可以用于多种场景。比如,在并发任务完成时通知主线程,或者控制并发任务的执行数量。例如,我们可以使用一个channel来等待后台排序任务的完成:

c := make(chan int)
go func() {
    list.Sort()
    c <- 1
}()
// 其他操作
<-c

这里,主线程在执行到<-c时会阻塞,直到接收到来自goroutine的信号,表明排序任务已完成。

并行化计算:充分利用多核CPU

利用goroutine和channel,我们可以轻松实现并行化计算。假设我们有一个需要对大量数据进行处理的任务,并且每个数据的处理是相互独立的。我们可以将数据分成多个部分,每个部分由一个goroutine来处理,然后通过channel来协调这些goroutine的执行。例如:

type Vector []float64

func (v Vector) DoSome(i, n int, u Vector, c chan int) {
    for ; i < n; i++ {
        v[i] += u.Op(v[i])
    }
    c <- 1
}

func (v Vector) DoAll(u Vector) {
    const numCPU = 4
    c := make(chan int, numCPU)
    for i := 0; i < numCPU; i++ {
        go v.DoSome(i*len(v)/numCPU, (i+1)*len(v)/numCPU, u, c)
    }
    for i := 0; i < numCPU; i++ {
        <-c
    }
}

在这个例子中,DoAll函数将数据分成numCPU个部分,分别由不同的goroutine进行处理。每个goroutine完成任务后,会通过channel发送一个信号,DoAll函数会等待所有的信号,确保所有任务都完成后才返回。

避免常见错误:数据竞争与资源管理

在并行编程中,要特别注意避免数据竞争和资源管理不当的问题。由于多个goroutine可能同时访问共享资源,数据竞争可能导致程序出现不可预测的行为。在Go语言中,通过使用channel来传递数据,而不是直接共享内存,可以有效避免数据竞争。同时,合理管理goroutine的数量和资源的使用也非常重要。例如,在处理大量请求时,要避免创建过多的goroutine导致资源耗尽。可以使用有缓冲的channel作为信号量来控制并发请求的数量:

var sem = make(chan int, MaxOutstanding)

func handle(r *Request) {
    sem <- 1
    process(r)
    <-sem
}

func Serve(queue chan *Request) {
    for {
        req := <-queue
        go handle(req)
    }
}

在这个例子中,sem通道的缓冲区大小限制了同时执行handle函数的数量,从而避免了资源的过度消耗。


http://www.kler.cn/a/522211.html

相关文章:

  • 【Julia】Julia预编译与外部库:从崩溃到完美集成
  • CMAKE工程编译好后自动把可执行文件传输到远程开发板
  • 性能优化2-删除无效引用
  • 获取snmp oid的小方法1(随手记)
  • 基于 AWS SageMaker 对 DeepSeek-R1-Distilled-Llama-8B 模型的精调与实践
  • 创作三载·福启新章2025
  • 【深度学习】搭建卷积神经网络并进行参数解读
  • ROS应用之SwarmSim在ROS 中的协同路径规划
  • obsidian插件——Metadata Hider
  • 软工_软件工程
  • Dest1ny漏洞库:用友 U8-CRM 系统 ajaxgetborrowdata.php 存在 SQL 注入漏洞
  • EtherCAT主站IGH-- 18 -- IGH之fsm_mbox_gateway.h/c文件解析
  • 使用Python Dotenv库管理环境变量
  • 日志收集Day008
  • 【系统架构设计师】操作系统 ① ( 知识的三种层次 - 系统知识、高频考点、试题拆解 - 软考备考策略 | 操作系统涉及的软考知识点 | 操作系统简介 )
  • 人机环境系统中的贝叶斯与非贝叶斯
  • 【算法学习笔记】36:中国剩余定理(Chinese Remainder Theorem)求解线性同余方程组
  • 06-机器学习-数据预处理
  • Vision Mamba在AMD GPU上使用ROCm
  • c语言版贪吃蛇(Pro Max版)附源代码
  • 题解 信息学奥赛一本通/AcWing 1118 分成互质组 DFS C++
  • 010 mybatis-PageHelper分页插件
  • 精通PCIe技术:协议解析与UVM验证实战
  • 大数据学习之SCALA分布式语言三
  • POWER SCHEDULER:一种与批次大小和token数量无关的学习率调度器
  • Mac Electron 应用签名(signature)和公证(notarization)