Skip to content

概述

原来分享基础语法的时候,还未分享过 chan 通道,这次把它补上。

chan 可以理解为队列,遵循先进先出的规则。

在说 chan 之前,咱们先说一下 go 关键字。

在 go 关键字后面加一个函数,就可以创建一个线程,函数可以为已经写好的函数,也可以是匿名函数。

为什么没有输出 goroutine ?

首先,我们清楚 Go 语言的线程是并发机制,不是并行机制。

那么,什么是并发,什么是并行?

并发是不同的代码块交替执行,也就是交替可以做不同的事情。

并行是不同的代码块同时执行,也就是同时可以做不同的事情。

举个生活化场景的例子:

你正在家看书,忽然电话来了,然后你接电话,通话完成后继续看书,这就是并发,看书和接电话交替做。

如果电话来了,你一边看书一遍接电话,这就是并行,看书和接电话一起做。

说回上面的例子,为什么没有输出 goroutine ?

main 函数是一个主线程,是因为主线程执行太快了,子线程还没来得及执行,所以看不到输出。

现在让主线程休眠 1 秒钟,再试试。

查看代码
go
package main

import (
	"fmt"
	"time"
)

func main()  {
	chan_1()
}

// chan通道 类似队列,先进先出
func chan_1()  {
	fmt.Println("主线程开始:", )
	go func() {
		fmt.Println("chan通道线程")
	}()
	// chan通道是并发的线程机制,而不是并行的线程机制
	// 通过主线程休眠让其他线程执行
	time.Sleep(1 * time.Second)
	fmt.Println("主线程结束:", )
}

输出:

main start
goroutine
main end

声明 chan

// 声明不带缓冲的通道
ch1 := make(chan string)

// 声明带10个缓冲的通道
ch2 := make(chan string, 10)

// 声明只读通道
ch3 := make(<-chan string)

// 声明只写通道
ch4 := make(chan<- string)

注意:

不带缓冲的通道,进和出都会阻塞。

带缓冲的通道,进一次长度 +1,出一次长度 -1,如果长度等于缓冲长度时,再进就会阻塞。

写入 chan

ch1 := make(chan string, 10)

ch1 <- "a"

读取 chan

val, ok := <- ch1
// 或
val := <- ch1

关闭 chan

close(chan)

注意:

  • close 以后不能再写入,写入会出现 panic
  • 重复 close 会出现 panic
  • 只读的 chan 不能 close
  • close 以后还可以读取数据

示例

定义的如果是一个无缓冲的 chan,赋值后就陷入了阻塞。 输出:

go
main start
fatal error: all goroutines are asleep - deadlock!

需要声明一个有缓冲的 chan。

输出:

查看代码
go
func chan_2()  {
	// 不带缓存的通道
	// ch_1 := make(chan string)
	// 无缓冲区的通道直接写入会造成死锁
	// ch_1 <- "data1"
	// 带10个缓存的通道
	ch_2 := make(chan string, 5)
	ch_2 <- "a" // 入 chan
	time.Sleep(1 * time.Second)
	go func() {
		// 从通道中取
		// val1 := <- ch_1
		// fmt.Println(ch_1, val1)
		// close(ch_1)
		val := <- ch_2 // 出 chan
		fmt.Println(val)
		}()

	// // 声明只读通道
	// ch_3 := make(<-chan string)
	// fmt.Println(ch_3)
	// // 声明只写通道
	// ch_4 := make(chan<- string)
	// fmt.Println(ch_4)
}

只读、只写 例子:

查看代码
go
func chan_3()  {
	fmt.Println("main start")
	// 声明只读通道
	// ch_3 := make(<-chan string, 10)
	// val_3 := <- ch_3 // 写入报错all goroutines are asleep - deadlock!
	go func() {
		// fmt.Println("只读通道: ", val_3)
	}()
	time.Sleep(1 * time.Second)
	fmt.Println("main end")
}
func chan_4()  {
	fmt.Println("main start")
	// 声明只写通道
	ch_4 := make(chan<- string, 10)
	ch_4 <- "a"
	ch_4 <- "a"
	close(ch_4)
	// ch_4 <- "a"// 关闭通道之后再写会报panic: send on closed channel
	go func() {
		// invalid operation: cannot receive from send-only channel ch_4
		// 不能读取只写通道
		// val_4 := <- ch_4 
		// fmt.Println("只写通道: ", val_4)
	}()
	time.Sleep(1 * time.Second)
	fmt.Println("main end")
}
查看代码
go
func chan_5()  {
	fmt.Println("main start")
	ch := make(chan string, 3)
	go producer(ch)
	go customer(ch) // 如果不提供消费者,通道就会阻塞

	time.Sleep(1 * time.Second)
	fmt.Println("main end")	
}

func producer(ch chan string) {
	fmt.Println("producer start")
	ch <- "a"
	ch <- "b"
	ch <- "c"
	ch <- "d"
	fmt.Println("producer end")
}

func customer(ch chan string) {
	for {
		msg := <- ch
		fmt.Println(msg)
	}
}

带缓冲的通道,如果长度等于缓冲长度时,再进就会阻塞, 输出:

main start
producer start
main end

如果同时开启线程取出数据,就不会被阻塞,输出:

main start
producer start
producer end
a
b
c
d
main end

Released under the MIT License.