今天带大家来学习一个特殊的golang语法。

我们看一下context的结构,会发现一个很新鲜的channel语法:

// A Context carries a deadline, cancelation signal, and request-scoped values
// across API boundaries. Its methods are safe for simultaneous use by multiple
// goroutines.
type Context interface {
    // Done returns a channel that is closed when this `Context` is canceled
    // or times out.
    Done() <-chan struct{}

    // Err indicates why this Context was canceled, after the Done channel
    // is closed.
    Err() error

    // Deadline returns the time when this Context will be canceled, if any.
    Deadline() (deadline time.Time, ok bool)

    // Value returns the value associated with key or nil if none.
    Value(key interface{}) interface{}
}

注意看Done() <- chan struct{},这个函数的声明怎么这么奇怪呢?下面来分解一下。

  • Done() chan struct{}:如果函数定义改成这样,其意义是,

    • 函数名Done,参数(),返回值chan struct{}。

    • 单独拿返回值来说,它是一个管道chan,内部的数据类型是struct{}。

    • 单独拿struct{}来说,我们熟悉type Name struct{a int, b bool}这样去定义一个结构体的类型,其实struct{...}就是定义结构体,和map[string]int这种定义是一样的,type只是给它启了一个别名。

  • <- chan struct{}:单独看这个表达式,我们知道如果ch := make(chan struct{}),那么<- ch是从管道里取出数据。但是chan struct{}是类型而不是变量,竟然能从一个类型里取数据??

其实<-chan int仍旧是一个管道类型,它叫做单向channel。如果是<-chan int,说明是只能读不能写的管道(也不能关闭),如果是chan <- int,说明是只能写不能读的管道(可以关闭),仅此而已!(扩展阅读)