Go面试题 | []int 能转换为 []interface 吗?

咱们结果先行,这个问题的答案是:不能。

如果你还想知道更多的信息,就往下看。^_^

有些时候我们希望有这样的写法:定义一个参数为 []interface 的函数,在程序运行的过程中,传入 []int 或其他类型的 slice,以此来达到少写一些代码的目的。譬如下面这个弱智的求 slice 和的例子:

package main

import "fmt"

func sliceSum(inters []interface{}) (res interface{}) {
 nums := inters.([]int)

 sum := 0
 for _, num := range nums {
  sum += num
 }

 return sum
}

func main() {
 is := []int{7, 8, 9, 10}

 fmt.Println(sliceSum(is))
}

为了把这个程序写得更通用一点,参数和返回值都是用的 interface 类型。编译,会报错:

./inter.go:6:16: invalid type assertion: inters.([]int) (non-interface type []interface {} on left)
./inter.go:19:22: cannot use is (type []int) as type []interface {} in argument to sliceSum

第一个错:不能将左边的 []interface{} 转换成右边的 []int,因为 []interface 本身并不是 interface 类型,所以不能进行断言。

第二个错:sliceSum 函数不能接受 []int 类型的参数,因为 []int 不是 []interface 类型。

先把程序改成正确的:

package main

import "fmt"

func sliceSum(inters []interface{}) (res interface{}){
 sum := 0
 for _, inter := range inters {
   sum += inter.(int)
 }

 return sum
}

func main() {
 is := []int{7, 8, 9, 10}
  
 iis := make([]interface{}, len(is))
 for i := 0; i < len(is); i++ {
   iis[i] = is[i]
 }

 fmt.Println(sliceSum(iis))
}

直接在循环的地方,对 inters 里的每个元素进行断言后再累加。

再来研究下 Go 官方说的:[]int[]interface{} 内存模型不一样是什么意思。

之前的 slice 文章讲过,slice 底层有 3 个属性:

773b3f77df4c2a30dfb41ad7a7e5a202.pngslice

interface文章讲过,interface 底层有两个属性:

d6aa03dc83ebe7071aa55c9f940b96c2.pnginterface

用 dlv 来调试,在关键地方打上断点:

db63f53ac4b482464e78680cbde81335.pngimage

知道了 slice 地址后,打印出该地址处的数据:

x -fmt hex -len 24 0xc000055f30

48d551cdfca67dbe36f061ddd9cd1add.pngint slice

第一行即 slice 底层的数组地址,0x04, 0x04 分别指的是长度、容量。0x07、0x08、0x09、0x0a 则是数组的四个元素。

bd86995b9241f35a677ccf1c3222da97.pngslice memory

同样的方法,来看看 interface slice 的内存布局:

3265eb72b0be3d983f23de5e08d783dc.pnginterface slice

其实也非常清楚,它的数据部分占 64 字节:因为一个 interface{} 占用 16 个字节,4 个元素所有是 64 个字节。

a8887db958f4b920af528feceea2cd9f.pnginterface memory

最后,总结一下:Go 官方规定,[]int 不能转换成 []interface{},因为两者是不同的类型,[]interface 不是 interface 类型,且两者的内存布局并不相同。

解决办法就是泛型。那泛型的原理是什么呢?又是怎么实现的呢?问就是不知道~😛

注:本文内容主要来自于 Eli 的博客[1]。

参考资料

[1]博客: https://eli.thegreenplace.net/2021/go-internals-invariance-and-memory-layout-of-slices/


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值