文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

[]int 能转换为 []interface 吗?

2024-12-14 01:33

关注

这个问题的答案是:不能。

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

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

  1. package main 
  2.  
  3. import "fmt" 
  4.  
  5. func sliceSum(inters []interface{}) (res interface{}) { 
  6.  nums := inters.([]int
  7.  
  8.  sum := 0 
  9.  for _, num := range nums { 
  10.   sum += num 
  11.  } 
  12.  
  13.  return sum 
  14.  
  15. func main() { 
  16.  is := []int{7, 8, 9, 10} 
  17.  
  18.  fmt.Println(sliceSum(is)) 

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

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

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

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

先把程序改成正确的:

  1. package main 
  2.  
  3. import "fmt" 
  4.  
  5. func sliceSum(inters []interface{}) (res interface{}){ 
  6.  sum := 0 
  7.  for _, inter := range inters { 
  8.    sum += inter.(int
  9.  } 
  10.  
  11.  return sum 
  12.  
  13. func main() { 
  14.  is := []int{7, 8, 9, 10} 
  15.    
  16.  iis := make([]interface{}, len(is)) 
  17.  for i := 0; i < len(is); i++ { 
  18.    iis[i] = is[i] 
  19.  } 
  20.  
  21.  fmt.Println(sliceSum(iis)) 

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

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

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

slice

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

interface

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

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

  1. x -fmt hex -len 24 0xc000055f30 

int slice

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

slice memory

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

interface slice

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

interface memory

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

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

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

参考资料

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

 

来源:码农桃花源内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯