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

相关推荐
Python私教11 分钟前
PyPika:Python SQL 查询构建器
开发语言·python·sql
全栈老实人_13 分钟前
考研互学互助系统|Java|SSM|VUE| 前后端分离
java·开发语言·tomcat·maven
天天进步201517 分钟前
Java全栈项目实战:校园报修服务系统
java·开发语言
Themberfue43 分钟前
Java 网络原理 ①-IO多路复用 || 自定义协议 || XML || JSON
xml·java·开发语言·网络·计算机网络·json
m0_6996595644 分钟前
DAY3 QT简易登陆界面优化
开发语言·qt·命令模式
励志成为大佬的小杨1 小时前
关键字初级学习
c语言·开发语言·算法
疯一样的码农1 小时前
基于Spring Boot + Vue3实现的在线商品竞拍管理系统源码+文档
java·spring boot·后端
Y编程小白2 小时前
SpringBoot的pom.xml文件中,scope标签有几种配置?
xml·spring boot·后端
szpc16212 小时前
100V宽压输入反激隔离电源,适用于N道沟MOSFET或GaN或5V栅极驱动器,无需光耦合
c语言·开发语言·人工智能·单片机·嵌入式硬件·生成对抗网络·fpga开发
m0_748234712 小时前
Java-33 深入浅出 Spring - FactoryBean 和 BeanFactory BeanPostProcessor
java·开发语言·spring