Go有无缓冲channel的区别

无缓冲的channel

channel的默认类型就是无缓冲的。当一个数据被发送到无缓冲的channel中,发送操作会被阻塞,知道有另一个goroutine从这个channel中接收这个数据。同样,当试图从一个无缓冲的channel中接收数据时,如果没有数据可以接收,接收操作也会被阻塞,直到有另一个goroutine发送数据到这个channel。因此,无缓冲的channel提高了一种强同步的通信机制,发送和接收操作在完成数据交换的时候都会阻塞,确保了数据在不同的goroutine之间精确同步。

有缓冲的channel

有缓冲的channel具有一个固定大小的缓冲区。当数据被发送到有缓冲的channel时,如果缓冲区未满,发送操作就会立刻返回,否则发送操作会阻塞,直到有另一个goroutinechannel中接收数据病空出空间。当从一个有缓冲的channel接收数据的时候,如果缓冲区有数据,接收操作会被立刻返回,否则接收操作会阻塞,直到有另一个goroutine发送数据到channel。因此,有缓冲的channel提供了一种弱同步的通信机制,发送和接收操作可能不会阻塞,是的goroutine可以继续执行其他的操作。

使用场景

  • 无缓冲的channel 适用于确保两个goroutine必须同步进行的场景,比如确保某个事件发生后才进行下一步操作。
  • 有缓冲的channel 适用于想要减少因等待而导致的阻塞的场景,或者当数据生产速度不定时,缓冲可以帮助平滑数据流动和处理。

以下是一个使用示例,无缓冲的情况下,发送操作会阻塞,因此打印操作也无法执行,直到接收操作完成;而有缓冲的情况下,发送操作不会阻塞,所以会继续向下执行打印操作。

go 复制代码
package main

import (
	"fmt"
	"time"
)

func main() {
	// 无缓冲的channel
	unbuffered := make(chan string)
	go func() {
		unbuffered <- "Hello, World!"
		fmt.Println("Sent message to unbuffered channel!")
	}()
	// 模拟处理延迟
	time.Sleep(3 * time.Second)
	fmt.Println(<-unbuffered)

	// 有缓冲的channel
	buffered := make(chan string, 1)
	go func() {
		buffered <- "Hello, World!"
		fmt.Println("Sent message to buffered channel!")
	}()
	// 模拟处理延迟
	time.Sleep(3 * time.Second)
	fmt.Println(<-buffered)
}

最后给大家推荐一个LinuxC/C++高级架构系统教程的学习资源与课程,可以帮助你有方向、更细致地学习C/C++后端开发,具体内容请见 https://xxetb.xetslk.com/s/1o04uB

相关推荐
审判长烧鸡7 小时前
Go命名规则【2】全场景命名避坑指南
go·命名规则·ai问答
蒸汽求职8 小时前
跨越 CRUD 内卷:半导体产业链与算力基建下的软件工程新生态
人工智能·科技·面试·职场和发展·软件工程·制造
小兵张健8 小时前
一场大概率没拿到 offer 的面试,让我更坚定去做喜欢的事
人工智能·面试·程序员
AI人工智能+电脑小能手10 小时前
【大白话说Java面试题】【Java基础篇】第7题:HashMap的get流程是什么
java·后端·面试·哈希算法·散列表·hash-index·hash
豹哥学前端12 小时前
别再背“var 提升,let/const 不提升”了:揭开暂时性死区的真实面目
前端·面试
何陋轩13 小时前
【重磅】悟空来了:国产AI编程助手深度测评,能否吊打Copilot?
人工智能·算法·面试
众少成多积小致巨14 小时前
Soong构建入门
android·go·编译器
小研说技术14 小时前
实时通信对比,一场MCP协议的技术革命
前端·后端·面试
ServBay14 小时前
2026年 Go 开发中没有它就不行的 10 个库
后端·go
山栀shanzhi15 小时前
C/C++之:构造函数为什么不能设置为虚函数?
开发语言·c++·面试