golang类型断言(Type Assertion)的应用

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
package main

import "fmt"

func main() {
add(1, 2)
add(int16(1), int16(2))
add(float32(1.1), float32(2.2))
add(float64(1.1), float64(2.2))
add(true, false)
}

func add(a, b interface{}) {
switch t := a.(type) {
case int:
fmt.Printf("type [%T] add res[%d]\n", t, a.(int)+b.(int))
case int16:
fmt.Printf("type [%T] add res[%d]\n", t, a.(int16)+b.(int16))
case float32:
fmt.Printf("type [%T] add res[%f]\n", t, a.(float32)+b.(float32))
case float64:
fmt.Printf("type [%T] add res[%f]\n", t, a.(float64)+b.(float64))
default:
fmt.Printf("type [%T] not support!\n", t)
}
}

输出结果:

type [int] add res[3]
type [int16] add res[3]
type [float32] add res[3.300000]
type [float64] add res[3.300000]
type [bool] not support!

用interface{}作参数,是不是很像C++的模板函数,而类型断言是不是很像C++的类层次间的下行转换(也是不一定成功的)。需要注意的是,a.(type)只能和switch搭配使用。在使用前得用断言指明变量的类型,如果断言错误就会触发panic。

如果不想触发panic,先做判断再使用。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
package main

import "fmt"

func main() {
a := int16(2)
b := int32(3)
add(a, b)
}

func add(a, b interface{}) {
_, ok := a.(int32)
if !ok {
fmt.Println("error type assertion!")
}
b = b
}

运行结果:
error type assertion!
2.用于转换结构体的interface{}类型字段
例如,我们写handler去接收消息,不可能每个发来的消息都写个函数去handle。利用空接口和类型断言的特性,就可以将业务抽象出来:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
package main

import "fmt"
import "time"

type NetMsg struct {
MsgID int16
Data interface{}
}

type Cat struct {
name string
age int16
}

type Dog struct {
name string
age int32
}

type human struct {
name string
age int64
}

func main() {
msg1 := NetMsg{1, Cat{"Qian", 1}}
msg2 := NetMsg{2, Dog{"doge", 8}}
msg3 := NetMsg{3, Dog{"allu", 18}}
msg_handler(msg1)
time.Sleep(2000 * time.Millisecond)
msg_handler(msg2)
time.Sleep(2000 * time.Millisecond)
msg_handler(msg3)
}

func msg_handler(msg NetMsg) {
switch msg.MsgID {
case 1:
cat := msg.Data.(Cat)
fmt.Printf("Do Something with Msg 1 %v \n", cat)
case 2:
dog := msg.Data.(Dog)
fmt.Printf("Do Something with Msg 2 %v \n", dog)
default:
fmt.Printf("Error MsgID [%d] \n", msg.MsgID)
}
}

运行结果:
Do Something with Msg 1 {Qian 1}
Do Something with Msg 2 {doge 8}
Error MsgID [3]

https://segmentfault.com/a/1190000012495480