go快速学习_Channel 管道

Channel的基本概念

Channal就是用来通信的,像Unix下的管道一样,
它的操作符是箭头" <-" , 箭头的指向就是数据的流向

ch <- v // 发送值v到Channel ch中
v := <-ch // 从Channel ch中接收数据,并将数据赋值给v

下面的程序演示了一个goroutine和主程序通信的例程。

package main

import "fmt"

func main() {
    //创建一个string类型的channel
    channel := make(chan string)

    //创建一个goroutine向channel里发一个字符串
    go func() { channel <- "hello" }()

    msg := <- channel
    fmt.Println(msg)
}

chan为先入先出的队列,有三种类型,双向,只读,只写,分别为"chan","chan<-","<-chan"
初始化时候,可以指定容量make(chanint,100);容量(capacity)代表Channel容纳的最多的元素的数量

Channel的阻塞

channel默认上是阻塞的,也就是说,如果Channel满了,就阻塞写,如果Channel空了,就阻塞读。于是,我们就可以使用这种特性来同步我们的发送和接收端。

package main

import "fmt"
import "time"

func main() {

    channel := make(chan string) //注意: buffer为1

    go func() {
        channel <- "hello"
        fmt.Println("write \"hello\" done!")

        channel <- "World" //Reader在Sleep,这里在阻塞
        fmt.Println("write \"World\" done!")

        fmt.Println("Write go sleep...")
        time.Sleep(3*time.Second)
        channel <- "channel"
        fmt.Println("write \"channel\" done!")
    }()

    time.Sleep(2*time.Second)
    fmt.Println("Reader Wake up...")

    msg := <-channel
    fmt.Println("Reader: ", msg)

    msg = <-channel
    fmt.Println("Reader: ", msg)

    msg = <-channel //Writer在Sleep,这里在阻塞
    fmt.Println("Reader: ", msg)
}

结果为

Reader Wake up...
Reader:  hello
write "hello" done!
write "World" done!
Write go sleep...
Reader:  World
write "channel" done!
Reader:  channel

总结

  • channel的定义
    • 定义语法 :
      • var ch =make(chan 通道中传递的数据类型,容量大小) 0为无缓冲,其余为有缓冲
    • 读channel:
      • <-ch 读到数据抛空,用来调节各个go程的先后顺序
      • num := <-ch 读到数据,存入num中
    • 写channel:
      • ch <- data data类型严格于定义的语法一致
    • 特性:
      1. 通道中的数据只能单向流动。一端读端,另一端必须写端。
      2. 通道中的数据只能一次读取,不能重复读。
      3. 读端和写端在不同的go程之间
      4. 读端读,写端不在写,读端阻塞。写端写,读端不在线,写端阻塞。
    • 系统3个特殊文件:
      • stdin:标准输入文件 ---键盘
      • stdout:标准输出文件 ---屏幕
      • stderr:标准错误文件 ---屏幕
  • channel分类
    • 无缓冲
      • 要求 读端写端同时在线,对端不在线,本端阻塞
    • 有缓冲
      • 读端,不在线,写端可以将数据直接写入缓冲区(不阻塞) 直到缓冲区被写满,依然没有读端,写端阻塞

go程间通信

  • 多个go程间,如果有多个共享资源时,需要分别同步

同步通信,异步通信

  • 同步通信:——无缓冲channel
    • 一个调用发出,如果没有得到结果,那么该调用不返回 ——阻塞
  • 异步通信:
    • 一个调用发出,不等待结果,直接返回——不阻塞

关闭channel

  • 当写端,写完数据后,使用close(ch)关闭channel

  • 读端,可以判断channel是否可读

  • for{
      if num,ok:= <-ch;ok{
        fmt.Println("num=",num)//读ch中的数据
      }else{
        break  //跳出数据
      }
    }
    // 读channel的内容,结束后自动跳出
    // 是上面一种的简单写法
    for num := range ch{
    }
    
  • 关闭channel的特性:

    • 如果写端没有关闭,暂停写入,读端阻塞等待
    • 如果写端已经关闭,不能写入,报错:==panic:send on closed channel==
    • 如果写端已经关闭,读端依然能读取,读到的是数据类型的零值

单向channel

  • 定义语法:
    • 双向channel:
      • var ch chan 数据类型
    • 单向读channel:
      • var chr <- chan int
      • chr = ch 双向channel给单向读channel赋值
      • 只能做读操作不能做写操作
    • 单向写channel:
      • var chw chan <- int
      • chw = ch 双向channel给单向写channel赋值
      • 只能做写操作不能做读操作
  • 单向channel==不能==给双向channel赋值
  • 单向channel的使用:
    • 主要应用于函数传参,单向channel可以在语法层面限定channel使用的方法
      • 单向读:不能写
      • 单向写:不能读

Select关键字

多个Channel的select

package main
import "time"
import "fmt"

func main() {
    //创建两个channel - c1 c2
    c1 := make(chan string)
    c2 := make(chan string)

    //创建两个goruntine来分别向这两个channel发送数据
    go func() {
        time.Sleep(time.Second * 1)
        c1 <- "Hello"
    }()
    go func() {
        time.Sleep(time.Second * 1)
        c2 <- "World"
    }()

    //使用select来侦听两个channel
    for i := 0; i < 2; i++ {
        select {
        case msg1 := <-c1:
            fmt.Println("received", msg1)
        case msg2 := <-c2:
            fmt.Println("received", msg2)
        }
    }
}

select的使用

  • 作用:用来监听channel上的数据流动
  • 特性:
    • 每一个case分支,都必须是IO操作
    • 通常将select置于for循环中
    • 一个case监听的channel不满足监听条件,当前case分支阻塞
    • 当所有case分支都不满足监听条件时,select如果有default分支时,走default,否则等待case满足
    • 当监听的多个case分支中,同时满足多个case,随机选择任意一个执行
    • 为防止忙轮询出现,可以适当选择省略default
  • break关键字不能应用于结束整个select,只能跳出一个分支
  • ==结论==:所有使用select的go程,与其他go程间,通信时,采用的是==异步通信==
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 218,525评论 6 507
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,203评论 3 395
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 164,862评论 0 354
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,728评论 1 294
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,743评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,590评论 1 305
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,330评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,244评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,693评论 1 314
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,885评论 3 336
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,001评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,723评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,343评论 3 330
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,919评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,042评论 1 270
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,191评论 3 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,955评论 2 355

推荐阅读更多精彩内容