分享
  1. 首页
  2. 文章

<8> go channel 通信通道

a11101171 · · 4037 次点击 · · 开始浏览
这是一个创建于 的文章,其中的信息可能已经有所发展或是发生改变。

go中最重要的一种通信通道就是channel
1.给一个 nil channel 发送数据,造成永远阻塞
2.从一个 nil channel 接收数据,造成永远阻塞
3.给一个已经关闭的 channel 发送数据,引起 panic
4.从一个已经关闭的 channel 接收数据,立即返回一个零值

package main
import "fmt"
// 此channel没有设置缓存,将被阻塞,所以都是执行default
func main() {
 messages := make(chan string)
 signals := make(chan bool)
 select {
 case msg := <-messages:
 fmt.Println("received message", msg)
 default:
 fmt.Println("no message received")
 }
 msg := "hi"
 select {
 case messages <- msg:
 fmt.Println("sent message", msg)
 default:
 fmt.Println("no message sent")
 }
 select {
 case msg := <-messages:
 fmt.Println("received message", msg)
 case sig := <-signals:
 fmt.Println("received signal", sig)
 default:
 fmt.Println("no activity")
 }
}
// 输出
// no message received
// no message sent
// no activity

Closing Channels.

package main
import "fmt"
// 这个例子在执行workder的工作时使用jobs管道来监听
// 当jobs关闭了,表示worker执行完毕,开始退出goroutine
// goroutine的退出又使用done来监听
func main() {
 jobs := make(chan int, 5)
 done := make(chan bool)
 go func() {
 for {
 j, more := <-jobs
 if more {
 fmt.Println("received job", j)
 } else {
 fmt.Println("received all jobs")
 done <- true
 return
 }
 }
 }()
 // 发送三个jobs worker执行
 for j := 1; j <= 3; j++ {
 jobs <- j
 fmt.Println("sent job", j)
 }
 close(jobs)
 fmt.Println("sent all jobs")
// worker被关闭
 <-done
}
// 输出
// sent job 1
// received job 1
// sent job 2
// received job 2
// sent job 3
// received job 3
// sent all jobs
// received all jobs

Range over Channels.

package main
import "fmt"
// 遍历channel 有两种方法:select和range
// select 在channel输入或输出才能走case
// range可以再channel 甚至close后都能遍历值
func main() {
 queue := make(chan string, 2)
 queue <- "one"
 queue <- "two"
 close(queue)
 for elem := range queue {
 fmt.Println(elem)
 }
}
// 输出
// one
// two

Timers

package main
import "time"
import "fmt"
// time.NewTimer 定时一次
// 说明time是阻塞输出
func main() {
 // 定时2s
 timer1 := time.NewTimer(time.Second * 2)
 // 阻塞输出
 <-timer1.C
 fmt.Println("Timer 1 expired")
 timer2 := time.NewTimer(time.Second)
 go func() {
 <-timer2.C
 fmt.Println("Timer 2 expired")
 }()
 stop2 := timer2.Stop()
 if stop2 {
 fmt.Println("Timer 2 stopped")
 }
}
// 输出
// Timer 1 expired
// Timer 2 stopped

Tickers

package main
import "time"
import "fmt"
// ticker 定时循环
func main() {
 ticker := time.NewTicker(time.Millisecond * 500)
 go func() {
 for t := range ticker.C {
 fmt.Println("Tick at", t)
 }
 }()
 time.Sleep(time.Millisecond * 1600)
 ticker.Stop()
 fmt.Println("Ticker stopped")
}
// 输出
// Tick at 2012年09月23日 11:29:56.487625 -0700 PDT
// Tick at 2012年09月23日 11:29:56.988063 -0700 PDT
// Tick at 2012年09月23日 11:29:57.488076 -0700 PDT
// Ticker stopped

原文:https://gobyexample.com/non-blocking-channel-operations


有疑问加站长微信联系(非本文作者)

本文来自:CSDN博客

感谢作者:a11101171

查看原文:<8> go channel 通信通道

入群交流(和以上内容无关):加入Go大咖交流群,或添加微信:liuxiaoyan-s 备注:入群;或加QQ群:692541889

关注微信
4037 次点击
添加一条新回复 (您需要 后才能回复 没有账号 ?)
  • 请尽量让自己的回复能够对别人有帮助
  • 支持 Markdown 格式, **粗体**、~~删除线~~、`单行代码`
  • 支持 @ 本站用户;支持表情(输入 : 提示),见 Emoji cheat sheet
  • 图片支持拖拽、截图粘贴等方式上传

用户登录

没有账号?注册
(追記) (追記ここまで)

今日阅读排行

    加载中
(追記) (追記ここまで)

一周阅读排行

    加载中

关注我

  • 扫码关注领全套学习资料 关注微信公众号
  • 加入 QQ 群:
    • 192706294(已满)
    • 731990104(已满)
    • 798786647(已满)
    • 729884609(已满)
    • 977810755(已满)
    • 815126783(已满)
    • 812540095(已满)
    • 1006366459(已满)
    • 692541889

  • 关注微信公众号
  • 加入微信群:liuxiaoyan-s,备注入群
  • 也欢迎加入知识星球 Go粉丝们(免费)

给该专栏投稿 写篇新文章

每篇文章有总共有 5 次投稿机会

收入到我管理的专栏 新建专栏