【go语言 | 第5篇】channel——多个goroutine之间通信

文章目录

channel的定义和使用

channel 用于多个 goroutine 之间的通信

go 复制代码
package main

import "fmt"

func main() {
	// 创建一个 channel
	c := make(chan int)

	go func() {
		defer fmt.Println("goroutine结束!")

		fmt.Println("goroutine正在运行")
		// 将数据发送到 channel 中
		c <- 100
	}()

	// 从 channel 中接收数据, 存入到 num
	num := <- c

	fmt.Println("num:", num)
	fmt.Println("main goroutine 结束")
}

为什么 main goroutine 中的执行会在 子goroutine 之后?

(1)main go 已经到达 num := <- c 时,如果 channel 中还没有存在 c,就会对 main go 进行阻塞。

(2)sub go 已经到达 c <- 100,如果 main go 还没有到达 num := <- c,因为此时的管道 c 是无缓冲的,就会对 sub go 进行阻塞。

channel------有缓冲和无缓冲同步

1. 无缓冲的channel

(1)在第1步,两个goroutine都到达通道,但哪个都没有开始执行发送或者接收。

(2)在第2步,左侧的goroutine将它的手伸进了通道,这模拟了向通道发送数据的行为。这时,这个goroutine会在通道中被锁住,直到交换完成。

(3)在第3步,右侧的goroutine将它的手放入通道,这模拟了从通道里接收数据。这个goroutine一样也会在通道中被锁住,直到交换完成。

(4)在第4步和第5步,进行交换,并最终,在第6步,两个goroutine都将它们的手从通道里拿出来,这模拟了被锁住的goroutine得到释放。两个goroutine现在都可以去做其他事情了。

2. 有缓冲的channel

go 复制代码
package main

import (
	"fmt"
	"time"
)

func main() {
	// 创建一个容量为3的 channel(有缓冲的channel)
	c := make(chan int, 3)

	go func() {
		defer fmt.Println("子go程结束!")

		for i := 0; i < 3; i++ {
			c <- i
			fmt.Println("子go程 i = ", i, "  元素个数:", len(c), "  元素个数:", cap(c))
		}	
	}()

	time.Sleep(1 * time.Second)

	for i := 0; i < 3; i++ {
		// 从 channel 中获取数据
		num := <- c
		fmt.Println("num =", num)
	}

	fmt.Println("main goroutine 运行结束")

}


(1)在第1步,右侧的goroutine正在从通道接收一个值。

(2)在第2步,右侧的这个goroutine独立完成了接收值的动作,而左侧的goroutine正在发送一个新值到通道里。

(3)在第3步,左侧的goroutine还在向通道发送新值,而右侧的goroutine正在从通道接收另外一个值。这个步骤里的两个操作既不是同步的,也不会互相阻塞。

(4)最后,在第4步,所有的发送和接收都完成,而通道里还有几个值,也有一些空间可以存更多的值。

channel------关闭channel

go 复制代码
package main

import "fmt"

func main() {
	c := make(chan int)

	go func() {
		for i := 0; i < 3; i++ {
			c <- i
		}

		// close 关闭 channel
		close(c)
	}()

	for {
		// ok 为 true 表示 channel 没有关闭, ok 为 false 表示 channel 已经关闭
		if data, ok := <- c; ok {
			fmt.Println("data: ", data) 
		} else {
			break
		}
	}

	fmt.Println("main goroutine 结束")

}

(1)channel不像文件一样需要经常去关闭,只有当你确实没有任何发送数据 了,或者想显式的结束range循环之类 的,才去关闭channel;

(2)关闭channel后,无法向channel再发送数据(引发panic错误后导致接收立即返回零值);

(3)关闭channel后,可以继续从channel接收数据;

(4)对于nil channel,无论收发都会被阻塞。

channel 与 range

go 复制代码
package main

import "fmt"

func main() {
	c := make(chan int)

	go func() {
		for i := 0; i < 3; i++ {
			c <- i
		}

		// close 关闭 channel
		close(c)
	}()

	/* for {
		// ok 为 true 表示 channel 没有关闭, ok 为 false 表示 channel 已经关闭
		if data, ok := <- c; ok {
			fmt.Println("data: ", data) 
		} else {
			break
		}
	} */

	// 可以使用 range 代替不断迭代的操作的cahnnel
	for data := range c {
		fmt.Println("data: ", data)
	}

	fmt.Println("main goroutine 结束")

}

channel 与 select

单流程下,一个go只能监控一个 channel 状态,select 可以完成监控多个 channel 的状态。

go 复制代码
package main

import (
	"fmt"
)

func method(c, quit chan int) {
	x, y := 1, 1

	for {
		select {
		case c <- x:
			// 如果 c 可写入,执行下面
			x = y
			y = x + y
		case <- quit:
			fmt.Println("quit.....")
			return 
		}
	}
}

func main() {
	c := make(chan int)
	quit := make(chan int)

	go func() {
		for i := 0; i < 6; i++ {
			fmt.Println(<-c)
		}

		quit <- 0
	}()

	method(c, quit)
}
相关推荐
HDO清风13 小时前
CASIA-HWDB2.x 数据集DGRL文件解析(python)
开发语言·人工智能·pytorch·python·目标检测·计算机视觉·restful
2201_7569890913 小时前
C++中的事件驱动编程
开发语言·c++·算法
多米Domi01113 小时前
0x3f 第48天 面向实习的八股背诵第五天 + 堆一题 背了JUC的题,java.util.Concurrency
开发语言·数据结构·python·算法·leetcode·面试
2301_8223776513 小时前
模板元编程调试方法
开发语言·c++·算法
csbysj202013 小时前
Python 循环嵌套
开发语言
测试_AI_一辰14 小时前
Agent & RAG 测试工程05:把 RAG 的检索过程跑清楚:chunk 是什么、怎么来的、怎么被命中的
开发语言·人工智能·功能测试·自动化·ai编程
Coding茶水间14 小时前
基于深度学习的输电电力设备检测系统演示与介绍(YOLOv12/v11/v8/v5模型+Pyqt5界面+训练代码+数据集)
开发语言·人工智能·深度学习·yolo·目标检测·机器学习
清风~徐~来14 小时前
【视频点播系统】BRpc 介绍及使用
开发语言
啟明起鸣14 小时前
【C++ 性能提升技巧】C++ 的引用、值类型、构造函数、移动语义与 noexcept 特性,可扩容的容器
开发语言·c++