本文为《Go专家编程》读书笔记~
协程A执行过程中需要创建子协程A1、A2、A3…An,协程A创建完子协程后就等待子协程退 出。针对这种场景,GO提供了三种解决方案:
程序通过创建N个channel来管理N个协程,每个协程都有一个channel用于跟父协程通信,父协程创建完所有协 程中等待所有协程结束。
channels[i] = make(chan int) //切片中放入一个channel
这个例子中,父协程仅仅是等待子协程结束,其实父协程也可以向管道中写入数据通知子协程结束,这时子协程需要定期的探测管道中是否有消息出现。
WaitGroup,可理解为Wait-Goroutine-Group,即等待一组goroutine结束。比如某个goroutine需要等待其 他几个goroutine全部完成,那么使用WaitGroup可以轻松实现。
var wg sync.WaitGroup
wg.Add(2) //设置计数器,数值即为goroutine的个数
----
wg.Done() //goroutine执行结束后将计数器减1
----
wg.Wait() //主goroutine阻塞等待计数器变为0
信号量是Unix系统提供的一种保护共享资源的机制,用于防止多个线程同时访问某个资源。
可简单理解为信号量为一个数值:
type WaitGroup struct {
state1 [3]uint32
}
state1是个长度为3的数组,其中包含了state和一个信号量,而state实际上是两个计数器:
Add()做了两件事,一是把delta值累加到counter中,因为delta可以为负值,也就是说counter有可能变成0或 负值,所以第二件事就是当counter值变为0时,跟据waiter数值释放等量的信号量,把等待的goroutine全部唤 醒,如果counter变为负值,则panic.
Wait()方法也做了两件事,一是累加waiter, 二是阻塞等待信号量
这里用到了CAS算法保证有多个goroutine同时执行Wait()时也能正确累加waiter。
Done()只做一件事,即把counter减1,我们知道Add()可以接受负值,所以Done实际上只是调用了Add(-1)。
它与WaitGroup最大的不同点是context对于派生 goroutine有更强的控制力,它可以控制多级的goroutine。
type Context interface {
Deadline() (deadline time.Time, ok bool)
Done() <-chan struct{
}
Err() error
Value(key interface{
}) interface{
}
}
该方法返回一个deadline和标识是否已设置deadline的bool值,如果没有设置deadline,则ok == false,此 时deadline为一个初始值的time.Time值
该方法返回一个channel,需要在select-case语句中使用,如”case <-context.Done():”。
当context关闭后,Done()返回一个被关闭的管道,关闭的管理仍然是可读的,据此goroutine可以收到关闭请求;当context还未关闭时,Done()返回nil
该方法描述context关闭的原因。
关闭原因由context实现控制,不需要用户设置。
比如Deadline context,关闭原因可能是因为deadline,也可能提前被主动关闭,那么关闭原因就会不同: 因deadline关闭:“context deadline exceeded”; 因主动关闭: “context canceled”。
当context关闭后,Err()返回context的关闭原因;当context还未关闭时,Err()返回nil;
有一种context,它不是用于控制呈树状分布的goroutine,而是用于在树状分布的goroutine间传递信息。 Value()方法就是用于此种类型的context,该方法根据key值查询map中的value。
children中记录了由此context派生的所有child,此context被cancle时会把其中的所有child都cancle掉。
cancelCtx与deadline和value无关,所以只需要实现Done()和Err()接口外露接口即可。
按照Context定义,Done()接口只需要返回一个channel即可。
对于cancelCtx来说只需要返回成员变量done即可。
按照Context定义,Err()只需要返回一个error告知context被关闭的原因。对于cancelCtx来说只需要返回成员 变量err即可。
cancel()内部方法是理解cancelCtx的最关键的方法,其作用是关闭自己和其后代,其后代存储在 cancelCtx.children的map中,其中key值即后代对象,value值并没有意义,这里使用map只是为了方便查询而 已。
WithCancel()方法作了三件事:
timerCtx在cancelCtx基础上增加了deadline用于标示自动cancel的最终时间,而timer就是一个触发自动 cancel的定时器。
对于接口来说,timerCtx在cancelCtx基础上还需要实现Deadline()和cancel()方法,其中cancel()方法是重写的。
Deadline()方法仅仅是返回timerCtx.deadline而矣。而timerCtx.deadline是WithDeadline()或 WithTimeout()方法设置的
cancel()方法基本继承cancelCtx,只需要额外把timer关闭。
timerCtx被关闭后,timerCtx.cancelCtx.err将会存储关闭原因
valueCtx只是在Context基础上增加了一个key-value对,用于在各级协程间传递一些数据。 由于valueCtx既不需要cancel,也不需要deadline,那么只需要实现Value()接口即可。
当前context查找不到key时,会向父节点查找(可以通过子context查询到父的value值。)
Go是静态类型语言,比如”int”、”float32”、”[]byte”等等。每个变量都有一个静态类型,且在编译 时就确定了。
type Myint int
var i int
var j Myint
i 和j 类型相同吗?A:i 和j类型是不同的。 二者拥有不同的静态类型,没有类型转换的话是不可以互相赋值 的,尽管二者底层类型是一样的。
interface类型是一种特殊的类型,它代表方法集合。 它可以存放任何实现了其方法的值。
接口类型的变量可以存储任何实现该接口的值
最特殊的interface类型为空interface类型,即 interface {} ,interface用来表示一组方法集合,所有实现该方法集合的类型都被认为是实现了该接口。那么空interface类型的方法集合为空,也就是说所有类型都可以认为是实现了该接口。
一个类型实现空interface并不重要,重要的是一个空interface类型变量可以存放所有值,记住是所有值,这才是最最重要的。 这也是有些人认为Go是动态类型的原因,这是个错觉。
interface类型的变量可以存放任何实现了该接口的值。还是以上面的 io.Reader
为例进行说 明, io.Reader
是一个接口类型, os.OpenFile()
方法返回一个 File 结构体类型变量,该结构体类型实现 了 io.Reader
的方法,那么 io.Reader
类型变量就可以用来接收该返回值。
var r io.Reader
tty, err := os.OpenFile("/dev/tty", os.O_RDWR, 0)
if err != nil {
return nil, err
}
r = tty
Q: r的类型是什么?A: r的类型始终是 io.Reader interface类型,无论其存储什么值。
Q:那 File 类型体现在哪里?A:r保存了一个(value, type)对来表示其所存储值的信息。 value即为r所持有元素的值,type即为所持有元素的底层类型
Q:如何将r转换成另一个类型结构体变量?比如转换成 io.Writer A:使用类型断言,如 w = r.(io.Writer) . 意思 是如果r所持有的元素如果同样实现了io.Writer接口,那么就把值传递给w。
前面之所以讲类型,是为了引出interface,之所以讲interface是想说interface类型有个(value,type)对, 而反射就是检查interface的这个(value, type)对的。具体一点说就是Go提供一组方法提取interface的 value,提供另一组方法提取interface的type.
反射包里有两个接口类型
var x float64 = 3.4
t := reflect.TypeOf(x) //t is reflext.Type
fmt.Println("type:", t)
v := reflect.ValueOf(x) //v is reflext.Value
fmt.Println("value:", v)
注意:反射是针对interface类型变量的,其中 TypeOf() 和 ValueOf() 接受的参数都是 interface{}
类型的,也即x值是被转成了interface传入
的。
对象x转换成反射对象v,v又通过Interface()接口转换成interface对象,interface对象通过.(float64)类型断言获取float64类型的值。
var x float64 = 3.4
v := reflect.ValueOf(x) //v is reflext.Value
var y float64 = v.Interface().(float64)
fmt.Println("value:", y)
通过反射可以将interface类型变量转换成反射对象,可以使用该反射对象设置其持有的值。
reflect.Value
提供了 Elem()
方法,可以获得指针向指向的 value 。
var x float64 = 3.4
v := reflect.ValueOf(&x)
// 传入reflect.ValueOf()函数的其实是x的值,而非x本身。即通过v修改其值是无法影响x的,也即是无效 的修改,所以golang会报错。
v.Elem().SetFloat(7.1)
// 获得指针向指向的 value
fmt.Println("x :", v.Elem().Interface())