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
复制代码

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


相关文章
|
7月前
|
存储 Go
Go语言之接口与多态 -《Go语言实战指南》
Go 语言中的接口是实现多态的核心机制,通过一组方法签名定义行为。任何类型只要实现接口的所有方法即视为实现该接口,无需显式声明。本文从接口定义、使用、底层机制、组合、动态行为到工厂模式全面解析其特性与应用,帮助理解 Go 的面向接口编程思想及注意事项(如 `nil` 陷阱)。
207 22
|
2月前
|
Java 编译器 Go
【Golang】(5)Go基础的进阶知识!带你认识迭代器与类型以及声明并使用接口与泛型!
好烦好烦好烦!你是否还在为弄不懂Go中的泛型和接口而烦恼?是否还在苦恼思考迭代器的运行方式和意义?本篇文章将带你了解Go的接口与泛型,还有迭代器的使用,附送类型断言的解释
168 3
|
7月前
|
存储 JSON Go
Go语言之空接口与类型断言
本文介绍了 Go 语言中空接口(`interface{}`)和类型断言的核心概念及其应用。空接口可存储任意类型数据,适用于通用函数、动态数据结构与 JSON 解析等场景;类型断言用于将接口变量还原为具体类型,推荐使用带 `ok` 的写法以避免程序崩溃。此外,文章通过示例讲解了 `type switch` 类型判断与 JSON 处理技巧,并总结了空接口的注意事项,强调滥用可能导致类型安全性降低。内容深入浅出,帮助开发者灵活运用这些特性。
193 15
|
12月前
|
Go 开发工具
百炼-千问模型通过openai接口构建assistant 等 go语言
由于阿里百炼平台通义千问大模型没有完善的go语言兼容openapi示例,并且官方答复assistant是不兼容openapi sdk的。 实际使用中发现是能够支持的,所以自己写了一个demo test示例,给大家做一个参考。
|
6月前
|
存储 JSON JavaScript
[go]byte类型, string 类型, json 类型
本文介绍了Go语言中byte类型的基本概念、特点及用法。byte是8位无符号整数,取值范围为0-255,常用于二进制数据操作,如网络通信和文件读写。文章还详细说明了byte与字符串的转换、遍历byte数据以及与其他类型间的转换。此外,探讨了Go中json.Marshal和json.Unmarshal函数实现[]byte与JSON间的转换,并对比了[]byte与JSON的区别,帮助开发者更好地理解其应用场景与差异。
230 2
|
7月前
|
Go
Go语言接口的定义与实现
Go 语言的接口提供了一种灵活的多态机制,支持隐式实现和抽象编程。本文介绍了接口的基本定义、实现方式、空接口的使用、类型断言以及接口组合等核心概念,并探讨了接口与 nil 的关系及应用场景。通过示例代码详细说明了如何利用接口提升代码的可扩展性和可测试性,总结了接口的关键特性及其在依赖注入、规范定义和多态调用中的重要作用。
286 14
|
7月前
|
设计模式 缓存 算法
Go如何进行高质量编程与性能调优实践
本文介绍了Go语言高质量编程与性能调优的实践方法。高质量编程包括良好的编码习惯(如清晰注释、命名规范)、代码风格与设计(如MVC模式)、简洁明了的代码原则,以及单元测试与代码重构的重要性。性能调优方面,涵盖算法优化、数据结构选择、I/O优化、内存管理、并行与并发处理优化及代码层面的改进。通过这些方法,可有效提升代码质量和系统性能。
152 13
|
7月前
|
算法 Go
Go语言模拟集合类型-《Go语言实战指南》
在 Go 语言中,虽然没有内建的集合(Set)类型,但可以通过 `map` 实现其功能。常用方式包括 `map[T]bool` 和更节省内存的 `map[T]struct{}`。前者以布尔值表示元素存在性,后者利用零内存开销的空结构体。文章介绍了集合的基本操作(添加、删除、判断、遍历),并通过封装示例展示如何创建自定义 Set 类型。这种实现方式适用于去重、唯一标记及集合运算等场景,简洁高效且易于扩展。
|
7月前
|
分布式计算 Go C++
初探Go语言RPC编程手法
总的来说,Go语言的RPC编程是一种强大的工具,让分布式计算变得简单如同本地计算。如果你还没有试过,不妨挑战一下这个新的编程领域,你可能会发现新的世界。
177 10
|
10月前
|
存储 算法 Go
Go语言实战:错误处理和panic_recover之自定义错误类型
本文深入探讨了Go语言中的错误处理和panic/recover机制,涵盖错误处理的基本概念、自定义错误类型的定义、panic和recover的工作原理及应用场景。通过具体代码示例介绍了如何定义自定义错误类型、检查和处理错误值,并使用panic和recover处理运行时错误。文章还讨论了错误处理在实际开发中的应用,如网络编程、文件操作和并发编程,并推荐了一些学习资源。最后展望了未来Go语言在错误处理方面的优化方向。
142 5