一 select的概念

Go语言中的 select 关键字,可以同时响应多个通道的操作,在多个管道中随机选择一个能走通的路!

select {
    case 操作1:
        响应操作1
    case 操作2:
        响应操作2
    ...
    default:
        没有操作的情况
}

如果有这样的需求,两个管道中只要有一个管道能够取出数据,那么就使用该数据:

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

func fn2(ch chan string) {
    time.Sleep(time.Second * 6)
    ch <- "fn2222"
}

func main() {

    ch1 := make(chan string)
    go fn1(ch1)

    ch2 := make(chan string)
    go fn2(ch2)

    select {
    case r1 := <-ch1:
        fmt.Println("r1=", r1)
    case r2 := <-ch2:
        fmt.Println("r2=", r2)
    }
}

由于fn1延迟较低,则就会一直得到fn1的数据。

二 select的一些注意事项

2.1 default

select支持default,如果select没有一条语句可以执行,即所有的通道都被阻塞,那么有两种可能的情况:

  • 如果给出了default语句,执行default语句,同时程序性的执行会从select语句后的语句中恢复
  • 如果没有default语句,那么select语句将被阻塞,直到至少有一个通信可以进行下去
  • 所以一般不写default语句

当然,在一些场景中(for循环使用select获取channel数据),如果channel被写满,也可能会执行default。

注意:select中的case必须是I/O操作。

2.3 channel超时解决

在并发编程的通信过程中,最需要处理的是超时问题,即向channel写数据时发现channel已满,或者取数据时发现channel为空,如果不正确处理这些情况,会导致goroutine锁死,例如:

// 如果永远没有人写入ch数据,那么上述代码一直无法获得数据,导致goroutine一直阻塞
i := <-ch

利用select()可以实现超时处理:

    timeout := make(chan bool, 1)

    go func() {
        time.Sleep(1e9)            // 等待1秒钟
        timeout <- true
    }()

    select {
        case <-ch:              // 能取到数据
        case <-timeout:         // 没有从-cha中取到数据,此时能从timeout中取得数据
    }

2.4 空select

空的select唯一的功能是阻塞代码:

    select {}

三 select的一些案例

3.1 案例一 模拟web开发

package main

import (
    "fmt"
)

/**
    模拟远程调用RPC:使用通道代替 Socket 实现 RPC 的过程。
    客户端与服务器运行在同 一个进程, 服务器和客户端在两个 goroutine 中运行。
 */

// 模拟客户端
func RPCClient(ch chan string, req string) (string, error) {
    ch <- req                            // 向服务器发送请求模拟:请求数据放入通道
    select {                            // 等待服务器返回模拟:使用select
    case data := <-ch:
        return data, nil
    }
}

// 模拟服务端
func RPCServer(ch chan string) {
    // 通过无限循环继续处理下一个客户端请求。
    for {
        data := <-ch
        fmt.Println("server received: ", data)
        ch <- "roger"                        // 向客户端反馈
    }
}

func main() {

    // 模拟 启动服务器
    ch := make(chan  string)
    go RPCServer(ch)

    // 模拟 发送数据
    receive, err := RPCClient(ch, "hi")
    if err != nil {
        fmt.Println(err)
    } else {
        fmt.Println("client receive: ", receive)
    }
}

3.2 案例二 使用通道晌应计时器的事件

Go语言中的 time 包提供了计时器的封装。由于 Go 语言中的通道和 goroutine 的设计, 定时任务可以在 goroutine 中以同步方式完成,也可以通过在 goroutine 中异步回调完成。

实现一段时间之后:

    exitCh := make(chan int)
    fmt.Println("start")

    time.AfterFunc(time.Second, func() {
        fmt.Println("1秒后,结束")
        exitCh <- 0
    })

    // 阻塞以等待结束
    <- exitCh

计时器与打点器:

  • 计时器( Timer)与倒计时类似,也是给定多少时间后触发, 创建后会返回 time.Timer 对象
  • 打点器( Ticker) 表示每到整点就会触发,创建后会返回 time.Ticker 对象

返回的时间对象内包含成员 C ,其类型是只能接收的时间通道 C<-chanTime ,使用这个通道就可以获得时间触发的通知。

示例:创建一个打点器, 每500毫秒触发一起:创建一个计时器, 2秒后触发,只触发一次。

    // 创建一个打点器,每500毫秒触发一次
    ticker := time.NewTicker(time.Millisecond * 500)

    // 创建一个计时器,2秒后触发
    stopper := time.NewTimer(time.Second * 2)

    // 声明计数变量
    var i int

    for {            // 不断检查通道情况
        select {
        case <-stopper.C:        // 计时器到了
            fmt.Println("stop")
            goto StopHere        // 跳出循环
        case <-ticker.C:        // 打点触发了
            i++                    // 记录触发多少次
            fmt.Println("tick", i)
        }
    }

    StopHere:
        fmt.Println("done")

results matching ""

    No results matching ""