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语言非常适合用于构建高性能的并发应用程序。

相关推荐
遥不可及~~斌28 分钟前
Spring Boot RESTful API 设计指南:查询接口规范与最佳实践
spring boot·后端·restful
papership29 分钟前
【入门级-C++程序设计:12、文件及基本读写-文件的基本概念&文本文件的基本操作】
开发语言·c++·青少年编程
SaleCoder1 小时前
用Python构建机器学习模型预测股票趋势:从数据到部署的实战指南
开发语言·python·机器学习·python股票预测·lstm股票模型·机器学习股票趋势
熟悉的新风景7 小时前
springboot项目或其他项目使用@Test测试项目接口配置-spring-boot-starter-test
java·spring boot·后端
玩代码7 小时前
备忘录设计模式
java·开发语言·设计模式·备忘录设计模式
技术猿188702783518 小时前
实现“micro 关键字搜索全覆盖商品”并通过 API 接口提供实时数据(一个方法)
开发语言·网络·python·深度学习·测试工具
放飞自我的Coder8 小时前
【colab 使用uv创建一个新的python版本运行】
开发语言·python·uv
晴空月明8 小时前
分布式系统高可用性设计 - 监控与日志系统
后端
艾莉丝努力练剑8 小时前
【数据结构与算法】数据结构初阶:详解顺序表和链表(四)——单链表(下)
c语言·开发语言·数据结构·学习·算法·链表
zyhomepage8 小时前
科技的成就(六十九)
开发语言·网络·人工智能·科技·内容运营