Go语言学习-channel

channel俗称管道,用于数据传递或数据共享,其本质是一个先进先出的队列,使用goroutine+channel进行数据通讯简单高效,同时也线程安全,多个goroutine可同时修改一个channel,不需要加锁。

channel可分为三种类型:

  • 只读channel:只能读channel里面数据,不可写入

  • 只写channel:只能写数据,不可读

  • 一般channel:可读可写

channel使用

定义和声明

var readOnlyChan <-chan int            // 只读chan
var writeOnlyChan chan<- int           // 只写chan
var mychan  chan int                     //读写channel
//定义完成以后需要make来分配内存空间,不然使用会deadlock
mychannel = make(chan int,10)//或者
read_only := make (<-chan int,10)//定义只读的channel
write_only := make (chan<- int,10)//定义只写的channel
read_write := make (chan int,10)//可同时读写

读写数据

需要注意的是:

  • 管道如果未关闭,在读取超时会则会引发deadlock异常

  • 管道如果关闭进行写入数据会pannic

  • 当管道中没有数据时候再行读取会读取到默认值,如int类型默认值是0

ch <- "wd"  //写数据
a := <- ch //读取数据
a, ok := <-ch  //优雅的读取数据

循环管道

需要注意的是:

  • 使用range循环管道,如果管道未关闭会引发deadlock错误

  • 如果采用for死循环已经关闭的管道,当管道没有数据时候,读取的数据会是管道的默认值,并且循环不会退出

package mainimport ("fmt""time"
)func main() {mychannel := make(chan int,10)for i := 0;i < 10;i++{mychannel <- i}close(mychannel)  //关闭管道fmt.Println("data lenght: ",len(mychannel))for  v := range mychannel {  //循环管道fmt.Println(v)}fmt.Printf("data lenght:  %d",len(mychannel))
}

带缓冲区channe和不带缓冲区channel

  • 带缓冲区channel:定义声明时候制定了缓冲区大小(长度),可以保存多个数据

-n不带缓冲区channel:只能存一个数据,并且只有当该数据被取出时候才能存下一个数据

ch := make(chan int) //不带缓冲区
ch := make(chan int ,10) //带缓冲区

不带缓冲区示例:

package mainimport "fmt"func test(c chan int) {for i := 0; i < 10; i++ {fmt.Println("send ", i)c <- i}
}
func main() {ch := make(chan int)go test(ch)for j := 0; j < 10; j++ {fmt.Println("get ", <-ch)}
}//结果:
send  0
send  1
get  0
get  1
send  2
send  3
get  2
get  3
send  4
send  5
get  4
get  5
send  6
send  7
get  6
get  7
send  8
send  9
get  8
get  9

channel实现作业池

我们创建三个channel,一个channel用于接受任务,一个channel用于保持结果,还有一个channel用于决定程序退出的时候。

package mainimport ("fmt"
)func Task(taskch, resch chan int, exitch chan bool) {defer func() {   //异常处理err := recover()if err != nil {fmt.Println("do task error:", err)return}}()for t := range taskch { //  处理任务fmt.Println("do task :", t)resch <- t //}exitch <- true //处理完发送退出信号
}func main() {taskch := make(chan int, 20) //任务管道resch := make(chan int, 20)  //结果管道exitch := make(chan bool, 5) //退出管道go func() {for i := 0; i < 10; i++ {taskch <- i}close(taskch)}()for i := 0; i < 5; i++ {  //启动5个goroutine做任务go Task(taskch, resch, exitch)}go func() { //等5个goroutine结束for i := 0; i < 5; i++ {<-exitch}close(resch)  //任务处理完成关闭结果管道,不然range报错close(exitch)  //关闭退出管道}()for res := range resch{  //打印结果fmt.Println("task res:",res)}
}

只读channel和只写channel

一般定义只读和只写的管道意义不大,更多时候我们可以在参数传递时候指明管道可读还是可写,即使当前管道是可读写的。

package mainimport ("fmt""time"
)//只能向chan里写数据
func send(c chan<- int) {for i := 0; i < 10; i++ {c <- i}
}
//只能取channel中的数据
func get(c <-chan int) {for i := range c {fmt.Println(i)}
}
func main() {c := make(chan int)go send(c)go get(c)time.Sleep(time.Second*1)
}
//结果
0
1
2
3
4
5
6
7
8
9

select-case实现非阻塞channel

原理通过select+case加入一组管道,当满足(这里说的满足意思是有数据可读或者可写)select中的某个case时候,那么该case返回,若都不满足case,则走default分支。

package mainimport ("fmt"
)func send(c chan int)  {for i :=1 ; i<10 ;i++  {c <-ifmt.Println("send data : ",i)}
}func main() {resch := make(chan int,20)strch := make(chan string,10)go send(resch)strch <- "wd"select {case a := <-resch:fmt.Println("get data : ", a)case b := <-strch:fmt.Println("get data : ", b)default:fmt.Println("no channel actvie")}}//结果:get data :  wd

channel频率控制

在对channel进行读写的时,go还提供了非常人性化的操作,那就是对读写的频率控制,通过time.Ticker实现。

示例:

package mainimport ("time""fmt"
)func main(){requests:= make(chan int ,5)for i:=1;i<5;i++{requests<-i}close(requests)limiter := time.Tick(time.Second*1)for req:=range requests{<-limiterfmt.Println("requets",req,time.Now()) //执行到这里,需要隔1秒才继续往下执行,time.Tick(timer)上面已定义}
}
//结果:
requets 1 2018-07-06 10:17:35.98056403 +0800 CST m=+1.004248763
requets 2 2018-07-06 10:17:36.978123472 +0800 CST m=+2.001798205
requets 3 2018-07-06 10:17:37.980869517 +0800 CST m=+3.004544250
requets 4 2018-07-06 10:17:38.976868836 +0800 CST m=+4.000533569


本文来自互联网用户投稿,文章观点仅代表作者本人,不代表本站立场,不承担相关法律责任。如若转载,请注明出处。 如若内容造成侵权/违法违规/事实不符,请点击【内容举报】进行投诉反馈!

相关文章

立即
投稿

微信公众账号

微信扫一扫加关注

返回
顶部