Go 中无缓冲通道与容量为1的缓冲通道的区别

作为学Go的菜鸟,之前我以为这两个应该是同一个东西,以为无缓冲通道是缓冲通道容量为1的一种特殊情况。然鹅,这俩货根本不是同一个东西。

无缓冲通道

无缓冲通道也称为同步通道,发送操作会阻塞,直到另一个 goroutine 在对应通道执行接收操作。

在单线程中,往无缓冲通道写入数据是会造成死锁的。

这里的A或B,无论谁先执行,谁都会阻塞以等待另一个goroutine执行,也就是说往里写得等来读的,从里读得等来写的。最重要的是,A和B对s1的读写是同步的,直观的理解是A和B对s1的读写是同时发生的,当A对s1写完了,则B从s1中就读完了。这样的特性可以用于做并发单位之间的同步操作,如果在A和B中对同一个无缓冲通道进行了读写,那么A和B一定会在读写的地方进行同步,谁先到谁阻塞等待另外一个。

package mainfunc main() {s1 := make(chan int)s1 <- 1 // A fatal error: all goroutines are asleep - deadlock!fmt.Println(<-s1)   // B fatal error: all goroutines are asleep - deadlock!
}

这里以一个具体的示例来表现无缓冲通道的用法和特性。使用 WaitGroup 避免主线程提前结束。

package mainimport ("fmt""sync""time"
)const Interval = 5func main() {fmt.Printf("%s: main thread start\n", time.Now().Format("2006-01-02 15:04:05"))naturals := make(chan int)var wg sync.WaitGroup// putwg.Add(1)go func() {defer wg.Done()for x := 0; x < Interval; x++ {naturals <- xfmt.Printf("%s: put %d \n", time.Now().Format("2006-01-02 15:04:05"), x)time.Sleep(time.Duration(x+1) * time.Second)}close(naturals)}()time.Sleep(Interval * time.Second) // wait sometime for writing// getwg.Add(1)go func() {defer wg.Done()for {x, ok := <-naturalsif !ok {fmt.Printf("%s: break", time.Now().Format("2006-01-02 15:04:05"))break // 通道关闭且读完}fmt.Printf("%s: get %d \n", time.Now().Format("2006-01-02 15:04:05"), x)}}()wg.Wait()
}
2024-04-09 22:40:43: main thread start
2024-04-09 22:40:48: get 0 
2024-04-09 22:40:48: put 0 
2024-04-09 22:40:49: put 1 
2024-04-09 22:40:49: get 1 
2024-04-09 22:40:51: put 2 
2024-04-09 22:40:51: get 2 
2024-04-09 22:40:54: put 3 
2024-04-09 22:40:54: get 3 
2024-04-09 22:40:58: put 4 
2024-04-09 22:40:58: get 4 
2024-04-09 22:41:03: break

我这块还把写入和获取的时间打印出来了,可以看到,针对同一个元素,往通道写入和从通道取出是同时完成的(get 和 put 对应的时间相同),这也是无缓冲通道为什么叫做同步通道的原因。

我在两个协程之间还特地 sleep 了几秒(因为有时间差,如果不同步的话,必然 put 操作早于 get 操作),但是从打印结果看出,put 写操作并没有先于 get 执行,而是同步执行。

容量为1的缓冲通道

对于容量为1的缓冲通道,只要从通道取值的时候,通道里有值,就是可以正常执行的。

package mainimport "fmt"func main() {s1 := make(chan int, 1)s1 <- 1fmt.Println(<-s1) // 1
}

我们可以尝试把上面例子中的无缓冲通道修改成容量为1的缓冲通道再运行试试,只是将

naturals := make(chan int) 修改为 naturals := make(chan int, 1)
package mainimport ("fmt""sync""time"
)const Interval = 5func main() {fmt.Printf("%s: main thread start\n", time.Now().Format("2006-01-02 15:04:05"))naturals := make(chan int, 1)var wg sync.WaitGroup// putwg.Add(1)go func() {defer wg.Done()for x := 0; x < Interval; x++ {naturals <- xfmt.Printf("%s: put %d \n", time.Now().Format("2006-01-02 15:04:05"), x)time.Sleep(time.Duration(x+1) * time.Second)}close(naturals)}()time.Sleep(Interval * time.Second) // wait sometime for writing// getwg.Add(1)go func() {defer wg.Done()for {x, ok := <-naturalsif !ok {fmt.Printf("%s: break", time.Now().Format("2006-01-02 15:04:05"))break // 通道关闭且读完}fmt.Printf("%s: get %d \n", time.Now().Format("2006-01-02 15:04:05"), x)}}()wg.Wait()
}

从打印结果可以看出,这里和无缓冲通道不同的地方在于,有缓冲的通道并不强制 channel 的读写者必须同时完成发送和接收,读操作只会在没有数据时阻塞,写操作只会在没有可用容量时阻塞,这就有点像阻塞队列了。  因此,本例中对于容量为1的缓冲通道,第一个数据写入后才开始阻塞然后等待读操作读取(因为读操作在 sleep 几秒之后才开始读取操作,通道已满,写操作暂时阻塞,待读操作读取数据后,写操作再继续执行)

2024-04-09 22:44:15: main thread start
2024-04-09 22:44:15: put 0 
2024-04-09 22:44:20: get 0 
2024-04-09 22:44:20: get 1 
2024-04-09 22:44:20: put 1 
2024-04-09 22:44:22: put 2 
2024-04-09 22:44:22: get 2 
2024-04-09 22:44:25: put 3 
2024-04-09 22:44:25: get 3 
2024-04-09 22:44:29: put 4 
2024-04-09 22:44:29: get 4 
2024-04-09 22:44:34: break

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/803119.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

TDengine Schemaless(无模式写入)常见问题的原因及故障排除

Tips&#xff1a;使用版本&#xff1a;3.0.2.6 &#xff08;一&#xff09;TDengine ERROR (80003002): Invalid data format 格式化问题&#xff1b;如缺少必要的组成格式&#xff08;时间戳、超级表等&#xff09;&#xff0c;或有字符串未作修饰符修饰&#xff0c;类似的还…

0基础如何进入IT行业?【模板】

0基础如何进入IT行业&#xff1f; 简介&#xff1a;对于没有任何相关背景知识的人来说&#xff0c;如何才能成功进入IT行业&#xff1f;是否有一些特定的方法或技巧可以帮助他们实现这一目标&#xff1f; 提醒&#xff1a;在发布作品前&#xff0c;请把不需要的内容删掉。 方…

工业通信原理——Modbus-RTU通信规约定义

工业通信原理——Modbus-RTU通信规约定义 前言 Modbus RTU是一种基于串行通信的通信协议&#xff0c;通常用于在设备之间进行数据通信。 Modbus-RTU通信规约定义 Modbus RTU通信规约的定义&#xff0c;包括客户机请求和服务器响应的基本流程&#xff1a; 物理层&#xff1…

c# wpf LiveCharts 饼图 简单试验

1.概要 c# wpf LiveCharts 饼图 简单试验 2.代码 <Window x:Class"WpfApp3.Window5"xmlns"http://schemas.microsoft.com/winfx/2006/xaml/presentation"xmlns:x"http://schemas.microsoft.com/winfx/2006/xaml"xmlns:d"http://schem…

javaScript中原型链

一、原型链 js 的对象分为普通对象和函数对象。每个对象都有__proto__ 但是只有函数对象 (非箭头函数) 才有 prototype 属性。 new的过程&#xff1a; 1、创建一个空的简单 javaScript对象 2、将空对象的 __proto__连接到该函数的 prototype 3、将函数的this指向新创建的对象…

ARP寻址过程

当知道目标的IP但是不知道目标的Mac地址的时候就需要借助ARP寻址获取目标的Mac地址&#xff0c;传输层借助四元组&#xff08;源IP源端口&#xff1a;目标IP目标端口&#xff09;匹配&#xff0c;网络层借助IP匹配&#xff0c;数据链路层则根据Mac地址匹配&#xff0c;数据传输…

RK3568---4G模块驱动实验

作者简介&#xff1a; 一个平凡而乐于分享的小比特&#xff0c;中南民族大学通信工程专业研究生在读&#xff0c;研究方向无线联邦学习 擅长领域&#xff1a;驱动开发&#xff0c;嵌入式软件开发&#xff0c;BSP开发 作者主页&#xff1a;一个平凡而乐于分享的小比特的个人主页…

基于keepalived+gtid+双vip半同步主从复制的MySQL高性能集群

项目名称&#xff1a;基于keepalivedgtid双vip半同步主从复制的MySQL高性能集群 目录 项目名称&#xff1a;基于keepalivedgtid双vip半同步主从复制的MySQL高性能集群 项目规划图 1.配置4台MySQL服务器&#xff08;1台master&#xff0c;2台slave&#xff0c;1台backup&a…

本地代码第一次提交到远程仓库gitee

1.在gitee新建仓库 2.新建一个空文件夹 打开黑窗口,执行命令 克隆仓库地址 执行命令 git clone https://gitee.com/llncomms/test.git打开隐藏的项目 复制全部内容到需要提交的代码中 3.在提交的代码中执行命令 $ git add .git commit -m 首次提交$ git push提交成功

php curl发送文件请求

快手开发者文档&#xff1a;https://open.kuaishou.com/platform/openApi?menu20 1、body参数 Content-Type为 multipart/form-data public function sendPostFileRequest1($url,$file_path){// 初始化cURL会话$ch curl_init();// 设置目标URLcurl_setopt($ch, CURLOPT_URL,…

ssm033单位人事管理系统+jsp

单位人事管理系统设计与实现 摘 要 现代经济快节奏发展以及不断完善升级的信息化技术&#xff0c;让传统数据信息的管理升级为软件存储&#xff0c;归纳&#xff0c;集中处理数据信息的管理方式。本单位人事管理系统就是在这样的大环境下诞生&#xff0c;其可以帮助管理者在短…

SpringMVC数据接收(全面/详细注释)

SpringMVC涉及组件&#xff1a; DispatcherServlet : SpringMVC提供&#xff0c;我们需要使用web.xml配置使其生效&#xff0c;它是整个流程处理的核心&#xff0c;所有请求都经过它的处理和分发&#xff01;[ CEO ]HandlerMapping : SpringMVC提供&#xff0c;我们需要进行…

从零开始:一步步学习爬虫技术的实用指南(一)

从零开始&#xff1a;一步步学习爬虫技术的实用指南&#xff08;一&#xff09; Urllib1.什么是互联网爬虫2.爬虫核心3.爬虫的用途4.爬虫的分类4.1 通用爬虫&#xff1a;4.1 聚焦爬虫&#xff1a; 5.反爬手段5.1 User‐Agent&#xff1a;5.2.代理IP5.3.验证码访问5.4.动态加载网…

11.python的字典dict(下) 遍历字典,结构优化

11.python的字典dict(下) 遍历所有的键值对 items()方法是字典的一个内置方法&#xff0c;用于返回字典中所有键值对的视图&#xff08;view&#xff09;。它返回一个可迭代的对象&#xff0c;每个元素都是一个包含键和对应值的元组。 下面用一个例子来说明items()方法的用法…

Vue中Suspense组件详细介绍

<Suspense> 是一个内置组件&#xff0c;用来在组件树中协调对异步依赖的处理。它让我们可以在组件树上层等待下层的多个嵌套异步依赖项解析完成&#xff0c;并可以在等待时渲染一个加载状态。 异步依赖​ 要了解 <Suspense> 所解决的问题和它是如何与异步依赖进行交…

服务器负载均衡原理及算法

服务器负载均衡原理及算法 一、引言 随着互联网技术的飞速发展&#xff0c;网络服务的需求日益增长&#xff0c;单台服务器的性能往往难以满足大规模并发访问的需求。因此&#xff0c;服务器负载均衡技术应运而生&#xff0c;它能够有效地将网络请求分发到多台服务器上&#…

[C++/Linux] UDP编程

一. UDP函数 UDP&#xff08;用户数据报协议&#xff0c;User Datagram Protocol&#xff09;是一种无连接的网络协议&#xff0c;用于在互联网上交换数据。它允许应用程序发送数据报给另一端的应用程序&#xff0c;但不保证数据报能成功到达&#xff0c;也就是说&#xff0c;它…

双指针2s总结

5.双指针 双指针理论基础 那么vector< char > 和 string 又有什么区别呢&#xff1f; 其实在基本操作上没有区别&#xff0c;但是 string提供更多的字符串处理的相关接口&#xff0c;例如string 重载了&#xff0c;而vector却没有。 所以想处理字符串&#xff0c;我们…

Java常用类(二)

常用类&#xff08;二&#xff09; Object类 超类&#xff0c;基类&#xff0c;所有类的直接或间接父类&#xff0c;位于继承树的最高层任何类&#xff0c;如没有书写extends显示继承某个类&#xff0c;都默认直接继承Object类&#xff0c;否则为间接继承Object类中所定义的方…

C语言题目:数组寻找最小绝对值

题目描述 输入10个数&#xff0c;找出其中绝对值最小的数&#xff0c;将它和最后一个数交换&#xff0c;然后输出这10个数。 输入格式 十个数 输出格式 交换后的十个数 样例输入 10 2 30 40 50 60 70 80 90 100 样例输出 10 100 30 40 50 60 70 80 90 2 代码解析 包含…