golang 使用select和channel实现协程通信、并发控制和超时控制

select是golang中的控制语句,和switch有点类似,但是使用场景和原理却是完全不同,使用select配合channel可以实现协程之间的通信,也可以实现io层面的超时控制,也可以实现对于并发的控制

一、语法特点

1、select中的每个case都必须是一个通道
2、多个case中的通道,哪个符合就执行哪个通道,如果没有没有符合的case,要看是否有设置了default,如果有设置了default,就执行default中的逻辑,如果没有设置default,就会一直阻塞至其中的case被执行

二、实现协程之间的通信

子协程向主协程通信

如下,main函数中定义了一个select,其中设置了一个通道的case,从ch通道中获取数据。
main函数执行到select会阻塞,直到获取通道的值,这时候通过子协程,对ch进行赋值,所以会先打印child,后打印select start

ch := make(chan int) 
    go func() {
        fmt.Println("child")
        time.Sleep(2 * time.Second)
        ch <- 1
    }()

    select {
    case res := <-ch:
        fmt.Println("select start")
        fmt.Println(res)
    }

输出如下


image.png
子协程向其他子协程通信

这边定义了一个子协程,对childCh通道写数据,另一个子协程在for无限循环中不断的获取childCh通道的值,如果有值,则打印得到的值

    childCh := make(chan int)
    go func() {
        for {
            select {
            case res := <-childCh:
                fmt.Println("child2")
                fmt.Println(res)
            }
        }
    }()
    go func() {
        fmt.Println("child")
        time.Sleep(2 * time.Second)
        childCh <- 1
    }()
    for {
        time.Sleep(time.Second)
    }

输出如下:


image.png
多个子协程向主函数通信

如下,定义了两个通道,select会在两个通道随机选择,哪个通道先准备好了数据,就执行哪个通道

ch1 := make(chan int)
    ch2 := make(chan int)
    go func() {
        fmt.Println("child2")
        ch2 <- 2
    }()
    go func() {
        fmt.Println("child1")
        ch1 <- 1
    }()

    for i := 0; i < 2; i++ {
        select {
        case res1 := <-ch1:
            fmt.Println("ch1")
            fmt.Println(res1)
        case res2 := <-ch2:
            fmt.Println("ch2")
            fmt.Println(res2)

        }
    }

以下,主函数main中定义了两个通道,select定义在for循环中,主函数不断的从ch1和ch2通道中获取数据,如果没有数据的话,则执行default,两个子协程则是不断的往通道中分别写入 from 1和from 2数据。

    // 定义两个通道
    ch1 := make(chan string)
    ch2 := make(chan string)

    // 启动两个 goroutine,分别从两个通道中获取数据
    go func() {
        for {
            ch1 <- "from 1"
        }
    }()
    go func() {
        for {
            ch2 <- "from 2"
        }
    }()

    // 使用 select 语句非阻塞地从两个通道中获取数据
    for {
        select {
        case msg1 := <-ch1:
            fmt.Println(msg1)
        case msg2 := <-ch2:
            fmt.Println(msg2)
        default:
            // 如果两个通道都没有可用的数据,则执行这里的语句
            fmt.Println("no message received")
        }
    }

三、实现io的超时控制

这边使用的是select+channel+time.After实现了超时控制,这边列的只是简单的demo,io逻辑可以类似如下代码,放在do函数中,当io请求时间超过了time.After定义的时间,就会输出timeout

package main

import (
    "fmt"
    "time"
)

func main() {
    // 创建一个可以取消的上下文

    // 定义两个通道
    ch := make(chan string)

    go do(ch)

    select {
    case res := <-ch:
        fmt.Println("res")
        fmt.Println(res)
    case <-time.After(time.Second * 2):
        fmt.Println("timeout")
    }

}

func do(ch chan string) {
    time.Sleep(3 * time.Second)
    ch <- "done"
}

四、实现并发控制

适用场景:

比如某个逻辑getHttp中的逻辑只能允许一定的并发执行,当并发过高的时候会拖垮其中的服务或者外部服务时,我们就需要对并发进行控制。

这边创建了一个最多只能存放2个值的semaphoreBig通道,当并发到来的时候,通过semaphoreBig 信号量进行控制只能同时处理2个并发,多余的并发会阻塞至有空余的信号量后才会执行。

package main

import (
    "context"
    "fmt"
    "github.com/gin-gonic/gin"
)

var semaphoreBig = make(chan struct{}, 2) // 并发控制信号量

func main() {
    r := gin.Default() //创建gin

    r.Use(func(c *gin.Context) {
        //处理请求
        c.Next()
    })
    r.GET("/", index) //绑定路由

    r.Run(":8001") //运行绑定端口
}

func index(c *gin.Context) {
    Do(context.TODO())
}

func Do(ctx context.Context) {

    select {
    case <-ctx.Done():
        fmt.Println("ctx done")
    default:
        var releaseSemaphore func()
        semaphoreBig <- struct{}{}
        releaseSemaphore = func() {
            <-semaphoreBig // 释放信号量
        }
        func() {
            defer releaseSemaphore()
            getHttp()
        }()
    }
}

// 执行相应的http逻辑
func getHttp() {
    fmt.Println("http")
}

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

推荐阅读更多精彩内容