Go入门26:select的几种用法

select的用法与switch非常类似,由select开始一个新的选择块,每个选择条件由case语句来描述。与switch语句可以选择任何可使用相等比较的条件相比,select有比较多的限制,其中最大的一条限制就是每个case语句里必须是一个IO操作,确切的说,应该是一个面向channel的IO操作。

在执行select语句的时候,运行时系统会自上而下地判断每个case中的发送或接收操作是否可以被立即执行(立即执行:意思是当前Goroutine不会因此操作而被阻塞)。

select语法格式

A "select" statement chooses which of a set of possible send or receive operations will proceed. It looks similar to a "switch" statement but with the cases all referring to communication operations.

select {

    caseSendStmt:

        //statements

    caseRecvStmt:

        //statements

    default:

        //statements

}

其中,

SendStmt : channelVariable <- value

RecvStmt : variable <-channelVariable

A case with a RecvStmt may assign the result of a RecvExpr to one or two variables, which may be declared using a short variable declaration(IdentifierList := value). The RecvExpr must be a (possibly parenthesized) receive operation(<-channelVariable). There can be at most one default case and it may appear anywhere in the list of cases.

示例:

ch1 := make(chan int, 1)

ch2 := make(chan int, 1)

ch1 <- 1

select {

    case e1 := <-ch1:

        //如果ch1通道成功读取数据,则执行该case处理语句

        fmt.Printf("1th case is selected. e1=%v", e1)

    case e2 := <-ch2:

        //如果ch2通道成功读取数据,则执行该case处理语句

        fmt.Printf("2th case is selected. e2=%v", e2)

    default:

        //如果上面case都没有成功,则进入default处理流程

        fmt.Println("default!.")

}

示例1:select语句会一直等待,直到某个case里的IO操作可以进行

func fun1(ch chan int) {

    time.Sleep(time.Second * 5)

    ch <- 1

}

func fun2(ch chan int) {

    time.Sleep(time.Second * 10)

    ch <- 1

}

func main() {

    var ch1 = make(chan int)

    var ch2 = make(chan int)

    go fun1(ch1)

    go fun2(ch2)

    select {

    case <-ch1:

        fmt.Println("The first case is selected.")

    case <-ch2:

        fmt.Println("The second case is selected.")

    }

}

编译运行:The first case is selected.

示例2:使用空值channel进行验证

所有跟在case关键字右边的发送语句或接收语句中的通道表达式和元素表达式都会先被求值。无论它们所在的case是否有可能被选择都会这样。 

求值顺序:自上而下、从左到右

//定义几个变量,其中chs和numbers分别代表通道列表和整数列表

var ch1 chan int

var ch2 chan int

var chs = []chan int{ch1, ch2}

var numbers = []int{1, 2, 3, 4, 5}

func main() {

    select {

    case getChan(0) <- getNumber(2):

        fmt.Println("1th case is selected.")

    case getChan(1) <- getNumber(3):

        fmt.Println("2th case is selected.")

    default:

        fmt.Println("default!.")

    }

}

func getNumber(i int) int {

    fmt.Printf("numbers[%d]\n", i)

    return numbers[i]

}

func getChan(i int) chan int {

    fmt.Printf("chs[%d]\n", i)

    return chs[i]

}

编译运行:

chs[0]
numbers[2]
chs[1]
numbers[3]
default!.

之所以输出default!,是因为chs[0]和chs[1]都是空值channel,和空值channel通信永远都不会成功。

示例3:使用非空值channel进行验证

//定义几个变量,其中chs和numbers分别代表通道列表和整数列表

var ch1 chan int = make(chan int, 1)  //声明并初始化channel变量

var ch2 chan int = make(chan int, 1)  //声明并初始化channel变量

var chs = []chan int{ch1, ch2}

var numbers = []int{1, 2, 3, 4, 5}

func main() {

    select {

    case getChan(0) <- getNumber(2):

        fmt.Println("1th case is selected.")

    case getChan(1) <- getNumber(3):

        fmt.Println("2th case is selected.")

    default:

        fmt.Println("default!.")

    }

}

func getNumber(i int) int {

    fmt.Printf("numbers[%d]\n", i)

    return numbers[i]

}

func getChan(i int) chan int {

    fmt.Printf("chs[%d]\n", i)

    return chs[i]

}

编译运行:

chs[0]
numbers[2]
chs[1]
numbers[3]
1th case is selected.

使用非空值channel进行IO操作,所以可以成功,没有走default分支。

示例4:如果有多个case同时可以运行,go会随机选择一个case执行

func main() {

    chanCap := 5

    ch := make(chan int, chanCap) //创建channel,容量为5

    for i := 0; i < chanCap; i++ { //通过for循环,向channel里填满数据

        select { //通过select随机的向channel里追加数据

        case ch <- 1:

        case ch <- 2:

        case ch <- 3:

        }

    }

    for i := 0; i < chanCap; i++ {

        fmt.Printf("%v\n", <-ch)

    }

}

编译运行:

2
1
2
1
1

示例5:使用break终止select语句的执行

func main() {

    var ch = make(chan int, 1)

    ch <- 1

    select {

    case <-ch:

        fmt.Println("This case is selected.")

        break //The following statement in this case will not execute.

        fmt.Println("After break statement")

    default:

        fmt.Println("This is the default case.")

    }

    fmt.Println("After select statement.")

}

编译运行:

This case is selected.

After select statement.

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,294评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,493评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,790评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,595评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,718评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,906评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,053评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,797评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,250评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,570评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,711评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,388评论 4 332
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,018评论 3 316
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,796评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,023评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,461评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,595评论 2 350

推荐阅读更多精彩内容

  • golang 的 select 的功能和select, poll, epoll相似, 就是监听 IO 操作,当 I...
    Uzero阅读 1,216评论 0 0
  • 01.{ 换行: Opening Brace Can't Be Placed on a Separate Lin...
    码农不器阅读 2,397评论 0 14
  • 并发基础 在说Golang的并发编程之前,先认识一下目前并发的几种实现方式: 1.多进程。操作系统实现的并发模型,...
    睡着别叫醒我阅读 2,221评论 0 1
  • fmt格式化字符串 格式:%[旗标][宽度][.精度][arg索引]动词旗标有以下几种:+: 对于数值类型总是输出...
    皮皮v阅读 1,090评论 0 3
  • 一部新近的老片儿 绝地逃亡 成龙主演 所以在电影院见到这个 貌似俄罗斯套娃般的他 很传神 看来电影院这个地方 以后...
    老伟漫眼阅读 189评论 0 2