Go 编程 | 连载 20 - 接口类型断言和转换

简介: Go 编程 | 连载 20 - 接口类型断言和转换

一、接口类型断言

Go 语言中使用接口断言将接口转换成另外一个接口或者另外一个类型,接口的转换在编码过程中非常常见。

类型断言的格式为:

// i:表示接口类型的变量
// T:转换的目标类型
// t:转换后的变量
t := i.(T)
复制代码

实现转换的基础是要求 i 变量要实现 T 接口的方法,如果没有完全实现 T 接口的方法,转换时则会引发宕机,因此可以通过两个值来接收 i.(T) 的返回,一个是转换后的变量 t,一个表示是 t 是是否完全实现 T 的方法,完全实现则为 true,否则为 false,为 false的情况下转换后的 t 为 0

t, ok := i.(T)
复制代码

接口类型断言及转换

实现某个接口的类型的同时实现了另一个接口,因此可以在两个接口间转换。

type Flyer interface {
   Fly()
}
type Fighter interface {
   Fight()
}
type Hero struct {
}
func (h *Hero) Fly(){
   fmt.Println("Hero: Fly")
}
func (h *Hero) Fight(){
   fmt.Println("Hero: Fight")
}
type Demon struct {
}
func (d *Demon) Fight(){
   fmt.Println("Demon: Fight")
}
复制代码
func main() {
   // 创建结构体指针类型
   IronManPtr := new(Hero)
   ThanosPtr := new(Demon)
   // 保存为接口类型变量
   m := map[string] interface{} {
      "IronMan": IronManPtr,
      "Thanos": IronManPtr,
   }
   // 遍历 m
   for name, obj := range m {
      // 接口断言转换,转换为 Fighter 接口
      fighter, isFighter := obj.(Fighter)
      // 转换为 Flyer 接口
      flyer, isFlyer := obj.(Flyer)
      fmt.Printf("Name: %v, isFighter: %v, isFly: %v\n", name, isFighter, isFlyer)
      if isFlyer {
         flyer.Fly()
      }
      if isFighter {
         fighter.Fight()
      }
   }
}
复制代码

执行上述代码,输出结果如下:

name: IronMan, isFighter: true, isFly: true
Hero: Fly
Hero: Fight
name: Thanos, isFighter: true, isFly: false
Demon: Fight
复制代码

上述代码中 IronManPtr 和 IronManPtr 两个结构体指针是存储在 Map 中的 interface{} 接口变量中,在遍历时转换为 Fighter 接口和 Flyer 接口。

Duck Type 既 鸭子类型,如果某个东西长得像鸭子,像鸭子一样游泳,像鸭子一样嘎嘎叫,那它就可以被看成是一只鸭子。

鸭子类型的含义就是忽略对象本身,专注于对象能够实现的功能,Fighter 接口有 Fight 功能,而存储在 interface{} 接口变量也实现了 Fight 功能或者 Fly 功能,因此可以认为它们是同一种类型,可以实现转换。

接口转换其他类型

在 main 函数中输入如下代码,将 Fighter 接口转换为 *Hero

func main() {
   // 创建结构体指针类型
   p1 := new(Hero)
   // Thanos := new(Demon)
   var fighter Fighter = p1
   // Fighter 接口 转换为 *Hero
   p2 := fighter.(*Hero)
   fmt.Printf("p1=%p\n", p1)
   fmt.Printf("p2=%p\n", p2)
}
复制代码

执行上述代码,输出结果如下:

p1=0x1164fc0
p2=0x1164fc0
复制代码

如果将 Fighter 接口转换为 *Demon 类型则会报错:

panic: interface conversion: main.Fighter is *main.Hero, not *main.Demon
goroutine 1 [running]:
main.main()
  /ex15.go:13 +0x2e
复制代码

这是因为转换时接口内保存的实例对应的类型指针, 须是要转换的对应的类型指针。


相关文章
|
21天前
|
存储 Rust Go
Go nil 空结构体 空接口有什么区别?
本文介绍了Go语言中的`nil`、空结构体和空接口的区别。`nil`是预定义的零值变量,适用于指针、管道等类型;空结构体大小为0,多个空结构体实例指向同一地址;空接口由`_type`和`data`字段组成,仅当两者均为`nil`时,空接口才为`nil`。
Go nil 空结构体 空接口有什么区别?
|
15天前
|
数据采集 监控 Java
go语言编程学习
【11月更文挑战第3天】
31 7
|
21天前
|
Unix Linux Go
go进阶编程:Golang中的文件与文件夹操作指南
本文详细介绍了Golang中文件与文件夹的基本操作,包括读取、写入、创建、删除和遍历等。通过示例代码展示了如何使用`os`和`io/ioutil`包进行文件操作,并强调了错误处理、权限控制和路径问题的重要性。适合初学者和有经验的开发者参考。
|
22天前
|
Go
go语言常量的类型
【10月更文挑战第20天】
23 2
|
29天前
|
Java 大数据 Go
Go语言:高效并发的编程新星
【10月更文挑战第21】Go语言:高效并发的编程新星
49 7
|
1月前
|
Go 数据处理 调度
Go语言中的并发模型:解锁高效并行编程的秘诀
本文将探讨Go语言中独特的并发模型及其在现代软件开发中的应用。通过深入分析 Goroutines 和 Channels,我们将揭示这一模型如何简化并行编程,提升应用性能,并改变开发者处理并发任务的方式。不同于传统多线程编程,Go的并发方法以其简洁性和高效性脱颖而出,为开发者提供了一种全新的编程范式。
|
2月前
|
存储 Go
Go to Learn Go之接口
Go to Learn Go之接口
31 7
|
1月前
|
并行计算 算法 搜索推荐
探索Go语言的高并发编程与性能优化
【10月更文挑战第10天】探索Go语言的高并发编程与性能优化
|
2月前
|
存储 缓存 Go
go语言编程系列(五)
go语言编程系列(五)
|
2月前
|
搜索推荐 Java 编译器
go语言编程系列(四)
go语言编程系列(四)