26.Go语言·管道Channel
2019-06-11 本文已影响0人
一枼落知天下
main.go
// Go语言·管道Channel
package main
import (
model "day31/model"
)
var content string = `
————————————————Go语言·管道Channel————————————————————
一、协程&管道
1.全局变量加锁[互斥锁],同步改进程序
sync包提供了基本的同步基元,
如互斥锁。除了Once和WaitGroup类型,
大部分都是适用于低水平程序线程,
高水平的同步使用channel通信更好一些。
二、管道Channel
1.就是一个数据结构-队列
2.数据先进先出 [FIFO,first in first out]
3.线程安全,多个goroutine访问时,不需要加锁,
不会产生资源竞争问题
channel本身就是线程安全的
4.channel有类型的
三、管道基本使用
var 变量名 chan 数据类型
var intChan chan int
var mapChan chan map[int][string]
var perChan chan Person
var perChan2 chan *Person
管道是引用类型
管道必须初始化才能写入数据,即make后才能使用
管道是有类型的 intChan只能写入整数int
四、注意事项:
1.只能放指定类型的数据
2.管道数据存满咯,不能继续放了(不能超过管道容量)
3.从管道中取出数据后,可以继续放入
4.在没有使用协程的情况下,如果管道数据取完了,再取,就会报dead lock
5.chan interface{} 需要类型断言interface--》Cat数据类型
比如:mycat.(Cat)
五、管道的遍历和关闭
1.使用内置函数close,一旦关闭就不能写入数据,只能读取
2.支持for-range的方式进行遍历:
a.遍历时,管道没有关闭,就会出现错误:
fatal error: all goroutines are asleep - deadlock!
b.遍历时,管道已经关闭,则会正常遍历,遍历完后,就会退出遍历
六、协程&管道双剑合璧
`
func main() {
model.Test()
}
model/Channel.go
package model
import (
"fmt"
)
type Cat struct {
Name string
Age int
}
/*
管道是引用类型
管道必须初始化才能写入数据,即make后才能使用
管道是有类型的 intChan只能写入整数int
*/
func Test() {
// 声明一个管道
var intChan chan int
// 初始化
intChan = make(chan int, 3)
// intChan的值:0xc00009c080,intChan本身的地址:0xc000096018
fmt.Printf("intChan的值:%v,intChan本身的地址:%p \n",
intChan, &intChan)
// 向管道写入数据
// 输入数据时,不能超过管道容量
intChan <- 10
num := 112
intChan <- num
intChan <- 98
// 看管道容量,长度
// intChan的长度:2,intChan本身的容量:3
fmt.Printf("intChan的长度:%v,intChan本身的容量:%v \n",
len(intChan), cap(intChan))
// 从管道中读取数据
// 读取数据时,也不能超过管道容量
var num2 int
num2 = <-intChan
// num2= 10
fmt.Println("num2=", num2)
// intChan的长度:2,intChan本身的容量:3
fmt.Printf("intChan的长度:%v,intChan本身的容量:%v \n",
len(intChan), cap(intChan))
}
/**
* [MiaoCat 任何数据类型的管道]
* @author Jhou Shuai
* @datetime 2019-06-11T16:24:59+0800
*/
func MiaoCat() {
var allChan chan interface{}
allChan = make(chan interface{}, 3)
allChan <- 10
allChan <- "ZhouShuai"
cat := Cat{"小喵喵", 4}
allChan <- cat
<-allChan
<-allChan
mycat := <-allChan
fmt.Printf("内容:%v,类型:%T \n", mycat, mycat)
// 下面的写法是错误的,编译不通过
// mycat.Name undefined (type interface {}
// is interface with no methods)
// fmt.Printf("mycat.Name = %v \n",mycat.Name)
// 类型断言interface--》Cat类型 mycat.(Cat)
fmt.Printf("mycat.Name = %v \n", mycat.(Cat).Name)
}
/**
* [CloseChannel 管道的关闭]
* @author Jhou Shuai
* @datetime 2019-06-11T16:37:06+0800
*/
func CloseChannel() {
intChan := make(chan int, 3)
intChan <- 100
intChan <- 10
// 内建函数close关闭管道
close(intChan)
// intChan<-23 //panic: send on closed channel
num1 := <-intChan
fmt.Println("num1=", num1)
}
func ForRangeChannel() {
intchan := make(chan int, 100)
for i := 0; i < 100; i++ {
intchan <- i * 2
}
close(intchan)
for val := range intchan {
fmt.Println(val)
}
}