Go语言使用goroutine及通道实现并发详解
目录
- 使用通道接收数据
- 阻塞接收数据
- 非阻塞接收数据
- 接收任意数据,忽略掉接收的数据
- 循环接收数据
使用通道接收数据
在上一篇文章中介绍了通道以及使用通道发送数据,本篇接着了解通道的基本内容,如何使用通道接收数据;
通道的接收同样使用"<-"操作符;
使用通道接收数据的特性如下:
- 通道的发送和接收操作在不同的两个goroutine间进行,由于通道中的数据在没有接收方接收时会持续阻塞,所以通道的接收必定在另外一个goroutine中进行;
- 如果在接收方接收时,通道中没有发送方发送数据,则接收方也会发生阻塞,直到发送方发送消息为止;
- 通道一次只能接收一个元素数据;
通道的数据接收有四种写法,接下来一一进行介绍
阻塞接收数据
阻塞模式在接收数据时,将接收变量作为"<-"操作符的左值,格式如下:
data := <-ch
执行该语句时将会阻塞,直到接收到数据并赋值给data变量
非阻塞接收数据
在使用非阻塞方式从通道接收数据时,语句不会发生阻塞,格式如下
data, ok := <-ch
data:接收到的数据,在未接收到数据时data为通道类型的零值
ok:是否接收到数据
非阻塞的通道接收方法,可能造成高的CPU占用,因此使用非常少,如果需要实现接收超时检测,则需要配合select和计时器进行;
接收任意数据,忽略掉接收的数据
写法如下:
<-ch
使用上面的这种写法,会发生阻塞,直到接收到消息,但是通道在接收到数据后会将其忽略掉;这个方式实际上只是通过通道在goroutine间阻塞收发,从而实现并发同步;
使用通道做并发同步的示例如下图所示:
循环接收数据
通道的数据接收可以借用for-range语句进行多个元素的接收操作,格式如下:
for data := range ch{ }
通道ch是可以被遍历的,遍历的结果就是接收到的数据,数据类型就是通道的数据类型,通过for遍历获得的变量只有一个,即上面例子中的data;
遍历通道数据的示例如下图所示:
通道可以在两个goroutine之间通过传递一个指定类型的值来同步运行和通讯,操作符"<-"用于指定通道的方向、发送和接收,如果未指定方向,则为双向通道;
ch <- v 把v发送到通道ch中
v := <-ch 从ch接收数据,并把值赋给v
下面列举一个示例通过两个goroutine计算数字之和,如下图所示:
以上就是通道接收数据的相关内容,更多关于Go goroutine通道并发的资料请关注我们其它相关文章!
相关推荐
-
Golang WorkerPool线程池并发模式示例详解
目录 正文 处理CVS文件记录 获取测试数据 线程池耗时差异 正文 Worker Pools 线程池是一种并发模式.该模式中维护了固定数量的多个工作器,这些工作器等待着管理者分配可并发执行的任务.该模式避免了短时间任务创建和销毁线程的代价. 在 golang 中,我们使用 goroutine 和 channel 来构建这种模式.工作器 worker 由一个 goroutine 定义,该 goroutine 通过 channel 获取数据. 处理CVS文件记录 接下来让我们通过一个例子,来进一步理
-
Go语言七篇入门教程四通道及Goroutine
目录 1. 前言 2. 通道简介 2.1 声明 2.1 读写 2.3 通道详解 2.3.1 例子 2.3.2 死锁 2.3.3 关闭通道 2.3.4 缓冲区 2.3.5 通道的长度和容量 2.3.6 单向通道 2.3.7 Select 2.3.8 default case 块 2.3.9 空 select 2.3.10 Deadlock 2.3.11 nil通道 2.4 多协程协同工作 2.5 WaitGroup 2.5.1 简介 2.5.2工作池 2.5.3 Mutex 3. 结语 如何学习G
-
Go并发同步Mutex典型易错使用场景
目录 Mutex的4种易错使用场景 1.Lock/Unlock 不成对出现 2.Copy 已使用的 Mutex 3.重入 4.死锁 解决策略 Mutex的4种易错使用场景 1.Lock/Unlock 不成对出现 Lock/Unlock 没有成对出现,就可能会出现死锁或者是因为Unlock一个未加锁的Mutex而导致 panic. 忘记Unlock的情形 代码中有太多的 if-else 分支,可能在某个分支中漏写了 Unlock: 在重构的时候把 Unlock 给删除了: Unlock 误写成了
-
GO中sync包自由控制并发示例详解
目录 资源竞争 sync.Mutex sync.RWMutex sync.WaitGroup sync.Once sync.Cond 资源竞争 channel 常用于并发通信,要保证并发安全,主要使用互斥锁.在并发的过程中,当一个内存被多个 goroutine 同时访问时,就会产生资源竞争的情况.这块内存也可以称为共享资源. 并发时对于共享资源必然会出现抢占资源的情况,如果是对某资源的统计,很可能就会导致结果错误.为保证只有一个协程拿到资源并操作它,可以引入互斥锁 sync.Mutex. syn
-
Go语言CSP并发模型goroutine channel底层实现原理
目录 Go的CSP并发模型(goroutine + channel) 1.goroutine goroutine的优点: 2.channel 无缓存channel 有缓存channel 3.Go并发模型的底层实现原理 4.一个CSP例子 参考Go的CSP并发模型实现:M, P, G Go语言是为并发而生的语言,Go语言是为数不多的在语言层面实现并发的语言. 并发(concurrency):多个任务在同一段时间内运行. 并行(parallellism):多个任务在同一时刻运行. Go的CSP并发模
-
Go语言并发编程基础上下文概念详解
目录 前言 1 Go 中的 Context 2 Context 接口 3 Context Tree 4 创建上下文 4.1 上下文创建函数 4.2 Context 使用规范 4.3 Context 使用场景 5 总结 前言 相信大家以前在做阅读理解的时候,一定有从老师那里学一个技巧或者从参考答案看个:结合上下文.根据上下文我们能够找到有助于解题的相关信息,也能更加了解段落的思想. 在开发过程中,也有这个上下文(Context)的概念,而且上下文也必不可少,缺少上下文,就不能获取完整的程序信息.那
-
Go语言使用goroutine及通道实现并发详解
目录 使用通道接收数据 阻塞接收数据 非阻塞接收数据 接收任意数据,忽略掉接收的数据 循环接收数据 使用通道接收数据 在上一篇文章中介绍了通道以及使用通道发送数据,本篇接着了解通道的基本内容,如何使用通道接收数据: 通道的接收同样使用"<-"操作符: 使用通道接收数据的特性如下: 通道的发送和接收操作在不同的两个goroutine间进行,由于通道中的数据在没有接收方接收时会持续阻塞,所以通道的接收必定在另外一个goroutine中进行: 如果在接收方接收时,通道中没有发送方发送数
-
Go语言学习教程之goroutine和通道的示例详解
目录 goroutine 通道 Range 和 Close Select 官方留的两道练习题 等价的二叉树 网络爬虫 源码地址 goroutine goroutine是由Go运行时管理的轻量级线程. go f(x, y, z)在一个新的goroutine中开始执行f(x, y,z). goroutines运行在相同的地址空间中,所以对共享的内存访问必须同步.sync包提供了基本的同步原语(synchronization primitives),比如互斥锁(mutual exclusion loc
-
Golang并发编程之main goroutine的创建与调度详解
目录 0. 简介 1. 创建main goroutine 2. 调度main goroutine 0. 简介 上一篇博客我们分析了调度器的初始化,这篇博客我们正式进入main函数及为其创建的goroutine的过程分析. 1. 创建main goroutine 接上文,在runtime/asm_amd64.s文件的runtime·rt0_go中,在执行完runtime.schedinit函数进行调度器的初始化后,就开始创建main goroutine了. // create a new goro
-
go语言context包功能及操作使用详解
目录 Context包到底是干嘛用的? context原理 什么时候应该使用 Context? 如何创建 Context? 主协程通知有子协程,子协程又有多个子协程 context核心接口 emptyCtx结构体 Backgroud TODO valueCtx结构体 WithValue向context添加值 Value向context取值 示例 WithCancel可取消的context cancelCtx结构体 WithDeadline-超时取消context WithTimeout-超时取消
-
Go语言实现定时器的原理及使用详解
目录 0. 前言 1. Timer定时器 1.1 time.NewTimer() 和 time.After() 1.2 time.AfterFunc 2. Ticker定时器 0. 前言 在进行并发编程时,有时候会需要定时功能,比如监控某个GO程是否会运行过长时间.定时打印日志等等. GO标准库中的定时器主要有两种,一种为Timer定时器,一种为Ticker定时器.Timer计时器使用一次后,就失效了,需要Reset()才能再次生效.而Ticker计时器会一直生效,接下来分别对两种进行介绍. 1
-
Go语言框架快速集成限流中间件详解
目录 前言 分布式版 简介 算法 实现 注意 单机版 简介 算法 实现 结语 前言 在我们的日常开发中, 常用的中间件有很多, 今天来讲一下怎么集成限流中间件, 它可以很好地用限制并发访问数来保护系统服务, 避免系统服务崩溃, 资源占用过大甚至服务器崩溃进而影响到其他应用! 分布式版 简介 通常我们的服务会同时存在多个进程, 也就是负载来保证服务的性能和稳定性, 那么就需要走一个统一的限流, 这个时候就需要借助我们的老朋友-redis, 来进行分布式限流; 算法 漏桶算法 即一个水桶, 进水(接
-
Go语言JSON解析器gjson使用方法详解
目录 gjson 安装 使用 gjson GJSON 是一个Go包,它提供了一种从json文档中获取值的快速简单的方法.它具有单行检索.点符号路径.迭代和解析 json 行等功能. 还可以查看SJSON以修改 json,以及JJ命令行工具. 本自述文件是如何使用 GJSON 的快速概述,有关更多信息,请查看GJSON 语法. github 的地址在这里. 安装 安装gjson,使用的是go传统的安装方法: go install github.com/tidwall/gjson@latest 在文
-
Go语言学习之context包的用法详解
目录 前言 需求一 需求二 Context 接口 emptyCtx valueCtx 类型定义 WithValue cancelCtx 类型定义 cancelCtx WithCancel timerCtx 类型定义 WithDeadline WithTimeout 总结 前言 日常 Go 开发中,Context 包是用的最多的一个了,几乎所有函数的第一个参数都是 ctx,那么我们为什么要传递 Context 呢,Context 又有哪些用法,底层实现是如何呢?相信你也一定会有探索的欲望,那么就跟
-
C语言中枚举与指针的实例详解
C语言中枚举与指针的实例详解 总结一下, 定义枚举,用typedef enum关键字, 比如 typedef enum{Red,Green,Blue} Color3; 枚举到数值的转换,如果没有指定代表数值就是从0开始算, 比如 Color3 c=Red; printf("%d",c);会显示0, 除非指定 如typedef enum{Red=3,Green=5,Blue=10} Color3; 关于类型指针的定义, 定义的时候在变量名左边加*代表此变量只是一个空指针而已, 若需要赋
-
C语言模拟实现atoi函数的实例详解
C语言模拟实现atoi函数的实例详解 atoi函数,主要功能是将一个字符串转变为整数,例如将"12345"–>12345.但在实现过程中,我们难免会因为考虑不够全面而漏掉比较重要的几点,今天就总结一下实现atoi函数需要注意的地方. 1.指针为NULL 2.字符串为空字符串 3.空白字符 4.正号与负号问题 5.溢出问题 6.异常字符处理 接下来看代码:(具体几种问题处理都在代码的注释中说明) #define _CRT_SECURE_NO_WARNINGS 1 #include
随机推荐
- PostgreSQL之分区表(partitioning)
- Python中用Spark模块的使用教程
- 深入解析Java编程中方法的参数传递
- Python命令行参数解析模块getopt使用实例
- asp.net中WebResponse 跨域访问实例代码
- asp.net如何进行mvc异步查询
- php 无限级分类,超级简单的无限级分类,支持输出树状图
- Python中threading模块join函数用法实例分析
- Android开发中使用sqlite实现新闻收藏和取消收藏的功能
- PHP生成随机数的方法实例分析
- php结合安卓客户端实现查询交互实例
- 如何让计数器只对新进用户计数?
- Linux命令集
- 微信小程序 教程之注册页面
- 给Linux新手[系列之四]
- Android 中 android.view.WindowLeaked的解决办法
- 详解离线安装npm包的几种方法
- 关于Mysql自增id的这些你可能还不知道
- 浅谈开发eslint规则
- JS栈stack类的实现与使用方法示例