优秀的编程知识分享平台

网站首页 > 技术文章 正文

Go语言是为并发而生

nanyue 2024-12-01 01:43:04 技术文章 4 ℃

作为程序员,要开发出能充分利用硬件资源的应用程序是一件很难的事情。现代计算机都拥有多个核,但是大部分编程语言都没有有效的工具让程序可以轻易利用这些资源。编程时需要写大量的线程同步代码来利用多个核,很容易导致错误。

Go语言正是在多核和网络化的时代背景下诞生的原生支持并发的编程语言。Go语言从底层原生支持并发,无须第三方库,开发人员可以很轻松地在编写程序时决定怎么使用 CPU 资源。

Go语言的并发是基于 goroutine 的,goroutine 类似于线程,但并非线程。可以将 goroutine 理解为一种虚拟线程。Go语言运行时会参与调度 goroutine,并将 goroutine 合理地分配到每个 CPU 中,最大限度地使用 CPU 性能。

多个 goroutine 中,Go语言使用通道(channel)进行通信,通道是一种内置的数据结构,可以让用户在不同的 goroutine 之间同步发送具有类型的消息。这让编程模型更倾向于在 goroutine 之间发送消息,而不是让多个 goroutine 争夺同一个数据的使用权。

程序可以将需要并发的环节设计为生产者模式和消费者的模式,将数据放入通道。通道另外一端的代码将这些数据进行并发计算并返回结果,如下图所示。



提示:Go语言通过通道可以实现多个 goroutine 之间内存共享。

【实例】生产者每秒生成一个字符串,并通过通道传给消费者,生产者使用两个 goroutine 并发运行,消费者在 main() 函数的 goroutine 中进行处理。

package main
import (
 "fmt"
 "math/rand"
 "time"
)
// 数据生产者
func producer(header string, channel chan<- string) {
 // 无限循环, 不停地生产数据
 for {
 // 将随机数和字符串格式化为字符串发送给通道
 channel <- fmt.Sprintf("%s: %v", header, rand.Int31())
 // 等待1秒
 time.Sleep(time.Second)
 }
}
// 数据消费者
func customer(channel <-chan string) {
 // 不停地获取数据
 for {
 // 从通道中取出数据, 此处会阻塞直到信道中返回数据
 message := <-channel
 // 打印数据
 fmt.Println(message)
 }
}
func main() {
 // 创建一个字符串类型的通道
 channel := make(chan string)
 // 创建producer()函数的并发goroutine
 go producer("cat", channel)
 go producer("dog", channel)
 // 数据消费函数
 customer(channel)
}

运行结果:

dog: 2019727887
cat: 1298498081
dog: 939984059
cat: 1427131847
cat: 911902081
dog: 1474941318
dog: 140954425
cat: 336122540
cat: 208240456
dog: 646203300

Tags:

最近发表
标签列表