实践一:
需求:
给Phone结构体增加一个特有方法 call() ,当Usb接口接收的是 Phone 变量的时候,还需要调用 call() 方法
代码
// 该案例展示 多态数组 类型断言package mainimport ("fmt")// 定义 Usb 接口, 有两个方法(行为)type Usb interface {start()stop()}// 定义 Phone 结构体, 并实现 start, stop 方法type Phone struct {Name string}// 值接收者// Phone 类型实现了 Usb 接口 (就是实现了 Usb 接口的所有方法)func (phone Phone) start() {fmt.Printf("%v开始--- \n", phone.Name)}func (phone Phone) stop() {fmt.Printf("%v停止--- \n", phone.Name)}// Phone 结构体特有方法func (phone Phone) call() {fmt.Printf("%v 手机打电话--- \n", phone.Name)}// 定义 Camera 结构体, 并实现 start, stop 方法type Camera struct {Name string}// 指针接收者// Camera 类型实现了 Usb 接口 (就是实现了 Usb 接口的所有方法)func (camera Camera) start() {fmt.Printf("%v开始--- \n", camera.Name)}func (camera Camera) stop() {fmt.Printf("%v停止--- \n", camera.Name)}// 定义 Computer 结构体, 并实现 Working 方法type Computer struct {}// 编写 Working 方法,接收一个 Usb 接口类型的变量,// 只要一个类型实现了某个接口,那么所有使用这个接口的地方,都支持这种类型的值// (比如: 实参 phone, camera, 都是 Usb 类型来接收, 这就是多态)func (computer *Computer) Working(usb Usb) {usb.start()// type Usb has no field or method call// usb.call() // err// 类型断言 调用 Phone.call()// 如果 usb 是指向 Phone 类型的变量,就把 usb 转化成 Phone 类型,并调用 Phone.call()if phone, ok := usb.(Phone); ok {phone.call()}usb.stop()}func main() {// 定义 Usb 接口数组var usbArr [3]Usbfmt.Println("usbArr = ", usbArr) // [<nil> <nil> <nil>]/*以下,接口类型的数组,存放了不同类型的元素(Phone, Camera...)这里就体现了多态数组*/usbArr[0] = Phone{"华为"}usbArr[1] = Camera{"尼康"}usbArr[2] = Phone{"苹果"}fmt.Println("usbArr = ", usbArr) // [{华为} 0xc00004c240 {苹果}]var computer Computerfor _, v := range usbArr {fmt.Println()computer.Working(v)}}/*usbArr = [<nil> <nil> <nil>]usbArr = [{华为} {尼康} {苹果}]华为开始---华为 手机打电话---华为停止---尼康开始---尼康停止---苹果开始---苹果 手机打电话---苹果停止---*/
实践二:
需求:
代码
// 编写函数,判断参数是什么类型package mainimport ("fmt")// 定义 Student 类型type Student struct{}// 可以接收多个不同类型的参数func JudgeType(items ...interface{}) {for i, v := range items {i = i + 1// 类型断言switch v.(type) {case bool:fmt.Printf("第%v个参数是 bool 类型,值:%v \n", i, v)case float32:fmt.Printf("第%v个参数是 float32 类型,值:%v \n", i, v)case float64:fmt.Printf("第%v个参数是 float64 类型,值:%v \n", i, v)case int, int16, int32, int64:fmt.Printf("第%v个参数是 整数 类型,值:%v \n", i, v)case string:fmt.Printf("第%v个参数是 string 类型,值:%v \n", i, v)case Student:fmt.Printf("第%v个参数是 Student 类型,值:%v \n", i, v)case *Student:fmt.Printf("第%v个参数是 *Student 类型,值:%v \n", i, v)default:fmt.Printf("第%v个参数类型不确定,值:%v \n", i, v)}}}func main() {var n1 float32 = 1.1var n2 float64 = 2.1var name string = "tom"address := "北京"n3 := 64stu1 := Student{}stu2 := &Student{}JudgeType(n1, n2, name, address, n3, stu1, stu2)var a interface{}var b *interface{}fmt.Printf("a 类型: %T, 值: %v \n", a, a) // a 类型: <nil>, 值: <nil>fmt.Printf("b 类型: %T, 值: %v \n", b, b) // b 类型: *interface {}, 值: <nil>}
