Go语言的并发与管道

Go语言以其简洁的语法和强大的并发处理能力而闻名。在Go中,goroutines是并发执行的基本单位,而channels则是goroutines之间通信的管道。本篇博客将详细介绍Go语言中的并发、管道(发送、接收)、无缓冲管道、有缓冲管道、关闭管道、单向管道以及多路复用的概念,并通过实例进行说明。

并发

并发是Go语言的核心特性之一,它允许我们同时执行多个任务。在Go中,goroutines是轻量级的线程,由Go运行时管理。goroutines之间的切换非常高效,因为它们共享同一个进程的内存空间。

示例

go 复制代码
package main

import (
	"fmt"
	"time"
)

func main() {
	go func() {
		fmt.Println("zhangsan")
	}()

	fmt.Println("我是main goroutine")
	time.Sleep(1 * time.Second)
}

在这个例子中,我们启动了一个goroutine,它会打印"zhangsan",而主线程会打印"我是main goroutine"。由于goroutine的执行是并发的,所以两个打印语句的执行顺序是不确定的。

管道(Channels)

管道是Go中goroutines之间通信的机制。通过管道,goroutines可以发送和接收数据。

发送和接收

发送数据到管道使用 <- 操作符,接收数据使用 <- 操作符。

示例

go 复制代码
package main

import (
	"fmt"
)

func main() {
	ch := make(chan string)
	go func() {
		fmt.Println("zhangsan")
		ch <- "goroutine 完成"
	}()

	fmt.Println("我是main goroutine")
	v := <-ch
	fmt.Println("接收到chan中的值:", v)
}

在这个例子中,一个goroutine发送了一个字符串到管道,主线程从管道中接收并打印了这个字符串。

无缓冲管道

无缓冲管道必须在接收者准备好之前,发送者不能发送数据,否则会阻塞。

示例

go 复制代码
package main

import (
	"fmt"
)

func main() {
	ch := make(chan string)
	go func() {
		ch <- "无缓冲管道"
	}()

	v := <-ch
	fmt.Println("接收到无缓冲管道中的值:", v)
}

在这个例子中,goroutine发送数据到无缓冲管道,主线程接收数据。如果主线程没有准备好接收,goroutine将会阻塞。

有缓冲管道

有缓冲管道可以在没有接收者的情况下存储一定数量的数据。

示例

go 复制代码
package main

import (
	"fmt"
)

func main() {
	cacheCh := make(chan int, 5)
	cacheCh <- 2
	cacheCh <- 3
	fmt.Println("cacheCh 容量为:", cap(cacheCh), "元素个数为:", len(cacheCh))
	go func() {
		fmt.Println(<-cacheCh)
	}()
}

在这个例子中,我们创建了一个容量为5的有缓冲管道。我们可以向管道中发送数据,而不会阻塞,直到管道满了为止。

关闭管道

关闭管道后,不能再向管道中发送数据,但可以接收数据。如果尝试向已关闭的管道发送数据,程序会panic。

示例

go 复制代码
package main

import (
	"fmt"
)

func main() {
	cacheCh := make(chan int, 5)
	cacheCh <- 2
	cacheCh <- 3
	close(cacheCh)
	go func() {
		fmt.Println(<-cacheCh)
	}()
}

在这个例子中,我们关闭了管道,然后启动了一个goroutine来接收数据。关闭管道后,不能再向管道中发送数据。

单向管道

单向管道只能用于发送或接收,不能同时进行。

示例

go 复制代码
package main

import (
	"fmt"
)

func main() {
	onlySend := make(chan<- int)
	onlyResive := make(<-chan int)
	onlySend <- 1
	<-onlyResive
}

在这个例子中,onlySend 是一个只发送的管道,onlyResive 是一个只接收的管道。尝试向只接收的管道发送数据或从只发送的管道接收数据都会导致编译错误。

多路复用

多路复用允许我们同时从多个管道中接收数据。

示例

go 复制代码
package main

import (
	"fmt"
)

func downLoad(chanName string) string {
	time.Sleep(1 * time.Second)
	return chanName + ":filePath"
}

func main() {
	firstCh := make(chan string)
	secondCh := make(chan string)
	threeCh := make(chan string)

	go func() {
		firstCh <- downLoad("first")
	}()
	go func() {
		secondCh <- downLoad("second")
	}()
	go func() {
		threeCh <- downLoad("three")
	}()

	select {
	case v := <-firstCh:
		fmt.Println("firstCh:", v)
	case v := <-secondCh:
		fmt.Println("secondCh:", v)
	case v := <-threeCh:
		fmt.Println("threeCh:", v)
	}
}

在这个例子中,我们启动了三个goroutine,每个goroutine下载一个文件并将其路径发送到不同的管道。主线程使用select语句从这三个管道中接收数据,实现了多路复用。

通过以上示例,我们可以看到Go语言在并发处理和管道通信方面的强大能力。这些特性使得Go语言非常适合用于构建高性能的并发应用程序。

相关推荐
应用市场5 小时前
构建自定义命令行工具 - 打造专属指令体
开发语言·windows·python
桦说编程5 小时前
深入解析CompletableFuture源码实现(2)———双源输入
java·后端·源码
舒一笑5 小时前
大模型时代的程序员成长悖论:如何在AI辅助下不失去竞争力
后端·程序员·掘金技术征文
lang201509285 小时前
Spring Boot优雅关闭全解析
java·spring boot·后端
Dfreedom.5 小时前
一文掌握Python四大核心数据结构:变量、结构体、类与枚举
开发语言·数据结构·python·变量·数据类型
一半烟火以谋生5 小时前
Python + Pytest + Allure 自动化测试报告教程
开发语言·python·pytest
虚行5 小时前
C#上位机工程师技能清单文档
开发语言·c#
小羊在睡觉6 小时前
golang定时器
开发语言·后端·golang
用户21411832636026 小时前
手把手教你在魔搭跑通 DeepSeek-OCR!光学压缩 + MoE 解码,97% 精度还省 10-20 倍 token
后端
追逐时光者6 小时前
一个基于 .NET 开源、功能强大的分布式微服务开发框架
后端·.net