go语言学习笔记 — 接口 — 接口与类型相互转换:在接口和类型之间转换

使用接口断言(type assertion),我们可以把接口转换成另外一个接口,也可以把接口转换成另外的类型。


1. 类型断言的格式

接口断言类似于控制流中的if,但大量类型断言出现时,应使用更高效的类型分支switch特性。

t := i.(T)

i表示接口变量(断言对象),T表示要转换的目标类型(指定类型),t表示转换后的变量(指定类型接口)。此时,如果i没有实现T,这个语句会触发panic。因此,有以下改进写法:

t, ok := i.(T)

如果断言对象是指定的类型,则返回指定类型接口;如果不是指定的类型,断言的第二个参数将返回false。

如果发生接口未实现,go将会把ok置为false,t置为T类型零值;正常实现时,ok为true。这里ok被认为是接口i是否实现类型T的结果。


2. 把接口转换成其他接口

鸟和猪具有不同特性,鸟可以飞,猪不能飞,但两种动物都可以走。如果使用结构体实现鸟和猪,让它们具备各自特性的Fly()和Walk()方法,就能让鸟和猪各自实现了飞行者接口(Flyer)和行走者接口(Walker)。

对保存有鸟或猪实例的空接口类型interface{}变量进行类型断言,如果断言对象是指定的类型,则返回指定类型接口;如果不是指定的类型,断言的第二个参数将返回false。

package main

import (
	"fmt"
	"testing"
)

// 接口定义
type Flyer interface { // 定义飞行动物接口
	Fly()
}

type Walker interface { // 定义行走动物接口
	Walk()
}

// 接口实现
type bird struct{} // 定义飞行类型

func (b *bird) Fly() { // 实现飞行动物飞行接口
	fmt.Println("bird: fly")
}

func (b *bird) Walk() { // 实现飞行动物行走接口
	fmt.Println("bird: walk")
}

// 接口实现
type pig struct{} // 定义行走类型

func (p *pig) Walk() { // 实现行走动物行走接口
	fmt.Println("pig: walk")
}

func main() {
	// 创建动物名到结构体实例的字典
	animals := map[string]interface{}{
		"bird": new(bird), // 创建出的结构体实例
		"pig":  new(pig),
	}

	// 遍历字典
	for name, obj := range animals { // obj为字典的值,是interface{}类型
		f, isFlyer := obj.(Flyer)   // 使用类型断言获得变量f,转换后的类型是Flyer;isFlyer是接口类型转换是否成功的结果
		w, isWalker := obj.(Walker) // 使用类型断言获得变量w,转换后的类型是Walker;isWalker是接口类型转换是否成功的结果

		fmt.Printf("name: %s isFlyer: %v isWalker: %v\n", name, isFlyer, isWalker)

		if isFlyer {
			f.Fly() // 调用接口方法
		}

		if isWalker {
			w.Walk() // 调用接口方法
		}
	}
}

3. 把接口转换成其他类型

在go中,接口和其他类型的自由转换,前提是接口已经实现。

把接口转换为普通的指针类型。例如,把Walker接口转换成*pig类型,

package main

import (
	"fmt"
)

// 接口定义
type Flyer interface { // 定义飞行动物接口
	Fly()
}

type Walker interface { // 定义行走动物接口
	Walk()
}

// 接口实现
type bird struct{} // 定义飞行类型

func (b *bird) Fly() { // 实现飞行动物飞行接口
	fmt.Println("bird: fly")
}

func (b *bird) Walk() { // 实现飞行动物行走接口
	fmt.Println("bird: walk")
}

// 接口实现
type pig struct{} // 定义行走类型

func (p *pig) Walk() { // 实现行走动物行走接口
	fmt.Println("pig: walk")
}

func main() {
	p1 := new(pig)

	var a Walker = p1
	p2 := a.(*pig) // 把Walker接口转换成*pig结构体指针类型

	fmt.Printf("p1=%p p2=%p", p1, p2) // 格式化输出指针类型
}

// p1=0x118efd0 p2=0x118efd0 ,对比发现p1和p2指针是相同的

在上述代码中,如果把Walker类型转换成*bird类型,将会报错:在接口转换时,main.Walker接口内部保存的是*main.pig,而不是*main.bird。因此,在接口转换成其他类型时,接口内保存实例对应的类型指针,必须要转换成相对应的类型指针。


  • 1
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值