1. Go断言的使用
Go中的断言用于判断变量的类型,其使用形式如下所示:
value, ok := x.(T)
上面的代码是判断x是否为T类型的变量:
truefalsetruefalsepanic
另外,断言和可以与switch配合使用
switch a.(type) { case int: fmt.Println("the type of a is int") case string: fmt.Println("the type of a is string") case float64: fmt.Println("the type of a is float") default: fmt.Println("unknown type") }
2. 闭包的解读
闭包是由函数和与其相关的引用环境组合而成的实体。
概念上说起来有些抽象,下面我们以一个具体的例子来理解。
func foo1(x int) func() { return func() { x = x + 1 fmt.Printf("foo2 val = %d\n", x) } } f1 := foo1(1) f1() // 2 f1() // 3
在上面的例子中,f1() 与他的变量x(值为1)共同组成了一个闭包,每次调用f1(),x的值就会+1并且打印。
从某种意义上来说,闭包延长了变量的生命周期(栈上分配改为了堆上分配)。
2.1 指针传递
func foo2(x *int) func() { return func() { *x = *x + 1 fmt.Printf("foo2 val = %d\n", *x) } } x := 1 f1 := foo2(&x) f2 := foo2(&x) f1() // 2 f2() // 3
通过第一个例子,我们知道,函数以及其环境(传入的变量)组成了闭包,这个时候,如果传入的是一个指针,那么就会存在多个闭包共用一个变量的情况。
2.2 延迟绑定
闭包的延迟绑定,通俗地说,就是闭包的函数在第一次调用的时候才会与环境的变量进行绑定,我们依然以上面提到的两个函数为例子:
func foo1(x int) func() { return func() { x = x + 1 fmt.Printf("foo2 val = %d\n", x) } } func foo2(x *int) func() { return func() { *x = *x + 1 fmt.Printf("foo2 val = %d\n", *x) } } x := 1 f1 := foo1(x) f2 := foo2(&x) f2() // 2 f1() // 3
- 我们创建了f1与f2两个闭包函数,以及变量 x 的值为1
- 在f1与f2创建的时候,变量并没有与函数绑定
- 第一次调用f2()时,&x与其绑定,x的值+1,变为2
- 第一次调用f1()时,x与其绑定,这时x已经变为2了,再+1,所以变为3
2.3 Go Routine的延迟绑定
我们在一个函数中启动 Go Routine 调用另一个函数:
func show(v interface{}) { fmt.Printf("foo4 val = %v\n", v) } func foo4() { values := []int{1, 2, 3, 5} for _, val := range values { go show(val) } } foo4() //foo3 val = 2 //foo3 val = 3 //foo3 val = 1 //foo3 val = 5
foo4()
但是,如果我们以匿名函数的形式尝试复现上面的逻辑,会发现:
func foo5() { values := []int{1, 2, 3, 5} for _, val := range values { go func() { fmt.Printf("foo5 val = %v\n", val) }() } } foo5() //foo3 val = 5 //foo3 val = 5 //foo3 val = 5 //foo3 val = 5
go func() { xxx }()