先来看看基本的定义:
channel是Go语言中的一个核心类型,可以把它看成管道。并发核心单元通过它就可以发送或者接收数据进行通讯,这在一定程度上又进一步降低了编程的难度。
channel是一个数据类型,主要用来解决go程的同步问题以及协程之间数据共享(数据传递)的问题。
(1)channle 本质上是一个数据结构——(队列),数据是先进先出。
(2)具有线程安全机制,多个go程访问时,不需要枷锁,也就是说channel本身是线程安全的。
(3)channel是有类型的,如一个string类型的channel只能存放string类型数据。
Channel遍历主要分为3种:
1)简单的读 data:=<-ch (如果读多次,需要用循环)
var ch8 = make(chan int, 6) func mm1() { for i := 0; i < 10; i++ { ch8 <- 8 * i } } func main() { go mm1() for i:=0;i<100;i++{ fmt.Print(<-ch8, "\t") } }
注:
(1)写入的次数与读取的次数需要一致(本例是10);
(2)如果读的次数多于写的次数会发生:fatal error: all goroutines are asleep - deadlock! ,若 在mm1中对ch8进行关闭(执行 close(ch8) ),多于的次数读到的数据为0(数据默认值)。
(3)读的次数少于写的次数,会读取出次数对应的内容,不会报错。
2)断言方式
if value, ok := <-ch; ok == true {
1) 如果写端没有写数据,也没有关闭。<-ch; 会阻塞 ---【重点】
2)如果写端写数据, value 保存 <-ch 读到的数据。 ok 被设置为 true
3)如果写端关闭。 value 为数据类型默认值。ok 被设置为 false
var ch8 = make(chan int, 6) func mm1() { for i := 0; i < 10; i++ { ch8 <- 8 * i } close(ch8) } func main() { go mm1() for { if data, ok := <-ch8; ok { fmt.Print(data,"\t") } else { break } } }
注:写完之后一定要关闭( 执行:close(ch8) ),否则会出现以下运行结果:
3)通过range方法
for num := range ch { }
var ch8 = make(chan int, 6) func mm1() { for i := 0; i < 10; i++ { ch8 <- 8 * i } close(ch8) } func main() { go mm1() for { for data := range ch8 { fmt.Print(data,"\t") } break } }
注:写完之后一定要关闭( 执行:close(ch8) ),否则会出现以下运行结果:
特别说明:以上实例都是子go程写,主go程读。如在子go程中写,另一个子go程中读,不管哪种方法,都不会出现以上错误问题。(多次实例验证)
var ch8 = make(chan int, 6) func mm1() { for i := 0; i < 10; i++ { ch8 <- 8 * i } //close(ch8) } func mm2() { for { for data:=range ch8{ fmt.Print(data,"\t") } } } func main() { go mm1() go mm2() for{ runtime.GC() } }
总结:
通过以上验证,为了保证程序的健壮性,在设计程序时,最好将channel的读、写分别在子go程中进行。写完数据之后,记得关闭channel。
补充一点:
channel不像文件一样需要经常去关闭,只有当你确实没有任何发送数据了,或者你想显式的结束range循环之类的,才去关闭channel;关闭channel后,无法向channel 再发送数据(引发 panic 错误后导致接收立即返回零值);关闭channel后,可以继续从channel接收数据;对于nil channel,无论收发都会被阻塞。
以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。如有错误或未考虑完全的地方,望不吝赐教。