Go高级并发模式

Go对并发提供了强大的原生支持,本文讨论Go的高级并发模式,理解这些并发模式,可以帮助我们编写高效的Go应用程序。原文: Advanced Concurrency Patterns in Go

"并发不是并行,但使并行成为可能。" ------ Rob Pike

本文将深入探讨Go中的一些高级并发模式。Go以其内置的并发原语而闻名,理解这些模式可以帮助我们编写更高效、可扩展的应用程序。


1. 基础Goroutine

goroutine是由Go运行时管理的轻量级线程。要启动一个goroutine,只需在函数前使用go关键字。

golang 复制代码
package main

import (
 "fmt"
 "time"
)

func sayHello() {
 fmt.Println("Hello from a goroutine!")
}

func main() {
 go sayHello() // This starts a new goroutine.
 time.Sleep(1 * time.Second) // Give goroutine some time to execute.
}

在本例中,sayHello函数与main函数并发运行。


2. Channel和Select

channel用于在程序之间进行通信,同步执行并确保数据安全。

基础channel示例

golang 复制代码
package main

import "fmt"

func main() {
 message := make(chan string) // create a new channel

 go func() { // start a goroutine
  message <- "Hello from the other side!" // send a message to the channel
 }()

 msg := <-message // receive a message from the channel
 fmt.Println(msg)
}

我们可以通过channel安全的在例程之间发送和接收消息。

使用Select

select允许程序等待多个通信操作,它就像一个针对channel的switch语句。

golang 复制代码
package main

import (
 "fmt"
 "time"
)

func main() {
 ch1 := make(chan string)
 ch2 := make(chan string)

 go func() {
  for {
   ch1 <- "from ch1"
   time.Sleep(2 * time.Second)
  }
 }()

 go func() {
  for {
   ch2 <- "from ch2"
   time.Sleep(3 * time.Second)
  }
 }()

 go func() {
  for {
   select {
   case msg1 := <-ch1:
    fmt.Println(msg1)
   case msg2 := <-ch2:
    fmt.Println(msg2)
   }
  }
 }()

 select {} // keep the main function alive
}

基于select,我们可以同时处理多个channel。


3. 高级模式: 工作池(Worker Pool)

工作池是一种限制运行的goroutine数量的方法。

工作池示例

golang 复制代码
package main

import (
 "fmt"
 "time"
)

func worker(id int, jobs <-chan int, results chan<- int) {
 for j := range jobs {
  fmt.Println("worker", id, "processing job", j)
  time.Sleep(time.Second)
  results <- j * 2
 }
}

func main() {
 const numJobs = 5
 jobs := make(chan int, numJobs)
 results := make(chan int, numJobs)

 // start 3 workers
 for w := 1; w <= 3; w++ {
  go worker(w, jobs, results)
 }

 // send jobs
 for j := 1; j <= numJobs; j++ {
  jobs <- j
 }
 close(jobs)

 // collect results
 for a := 1; a <= numJobs; a++ {
  <-results
 }
}

工作池帮助我们管理和限制并发运行的goroutine数量。


结论

Go中的并发(goroutine、channel和模式)为开发人员提供了强大的工具集。通过理解和利用这些概念,可以构建高性能和可伸缩的应用程序。


你好,我是俞凡,在Motorola做过研发,现在在Mavenir做技术工作,对通信、网络、后端架构、云原生、DevOps、CICD、区块链、AI等技术始终保持着浓厚的兴趣,平时喜欢阅读、思考,相信持续学习、终身成长,欢迎一起交流学习。为了方便大家以后能第一时间看到文章,请朋友们关注公众号"DeepNoMind",并设个星标吧,如果能一键三连(转发、点赞、在看),则能给我带来更多的支持和动力,激励我持续写下去,和大家共同成长进步!

本文由mdnice多平台发布

相关推荐
tyung1 天前
Go 手写有界 SPSC 环形队列:无 CAS、无锁、Cache 友好的无锁模型
后端·go
喵个咪2 天前
技术复盘:基于 go-wind-cms 的官网+商城双业务渐进拆分实战
后端·架构·go
止语Lab2 天前
Go context 超时传播:你以为设了就安全了
go
踏着七彩祥云的小丑3 天前
Go学习第9天:并发编程 + 文件操作 + 正则表达式
学习·golang·正则表达式·go
止语Lab3 天前
Go 代码生成的三层认知:从忍住不用到自己造轮子
go
协享科技3 天前
AI 视频理解:让 Agent 看视频并总结内容
人工智能·go·音视频·agent·ai编程
曲幽4 天前
掏出手机就能搭个 WebDAV 同步服务器?这操作有点香
go·termux·tampermonkey·sync·webdav·filebrowser·gowebdav·koreader
Code_Artist4 天前
🦜用 GoAI 从零打造一个 AI Agent 脚手架工程:重新定义智能体开发范式!
go·agent·ai编程
ShuiShenHuoLe5 天前
OS的常用函数
go
踏着七彩祥云的小丑5 天前
Go学习第8天:接口 + 泛型 + 错误处理
开发语言·学习·golang·go