当前位置:首页 > 科技  > 软件

Go 面试中的隐藏陷阱:SliceHeader 问题解析

来源: 责编: 时间:2024-06-12 17:31:35 89观看
导读大家好,我是煎鱼。最近也是面试季+毕业季了,很多同学正在积极准备面试。尤其是很多同学,已经通过官网资料熟悉了 Go 基本语法,但没有太大把握。希望对一些常见的棘手面试问题做一些预习。今天和大家学习 @Harutyun Mardir

大家好,我是煎鱼。1Sd28资讯网——每日最新资讯28at.com

最近也是面试季+毕业季了,很多同学正在积极准备面试。尤其是很多同学,已经通过官网资料熟悉了 Go 基本语法,但没有太大把握。希望对一些常见的棘手面试问题做一些预习。1Sd28资讯网——每日最新资讯28at.com

今天和大家学习 @Harutyun Mardirossian 大佬分享的面试题,一起进步!1Sd28资讯网——每日最新资讯28at.com

面试问题

请先在脑子里思考一下具体的运行结果,再查看答案。1Sd28资讯网——每日最新资讯28at.com

如下代码:1Sd28资讯网——每日最新资讯28at.com

func main() {    s := make([]int, 0, 2)    doSomething(s)    fmt.Println(s)}func doSomething(a []int) {    a = append(a, 1)}

面试问题:fmt.Println 的输出结果是什么?1Sd28资讯网——每日最新资讯28at.com

问题解析

运行程序,查看输出结果:1Sd28资讯网——每日最新资讯28at.com

[]

fmt.Println 最终打印的是一个长度为 0 的切片。1Sd28资讯网——每日最新资讯28at.com

答案是:空切片。(你答对了吗?)1Sd28资讯网——每日最新资讯28at.com

在 Go 中,函数参数是按值传递的,这意味着上述代码在参数传递时,创建了参数值的副本并传递给函数。1Sd28资讯网——每日最新资讯28at.com

而切片实际上是一个包含长度(len)、容量(cap)和指向底层数组指针(data)的结构体。1Sd28资讯网——每日最新资讯28at.com

当我们将切片作为函数参数传递时,实质上复制的是切片的 SliceHeader,对应的底层数组是保持不变的。1Sd28资讯网——每日最新资讯28at.com

结合代码来讲,就是因为在 doSomething 函数中,创建了 SliceHeader 的新副本。然后 append 函数会在超过容量时重新分配新切片,并返回更新后的切片。1Sd28资讯网——每日最新资讯28at.com

深入验证

我们可以使用 unsafe 包去打印 SliceHeader(切片头),进行进一步的验证和分析。1Sd28资讯网——每日最新资讯28at.com

如下代码:1Sd28资讯网——每日最新资讯28at.com

type SliceHeader struct {    Data uintptr    Len  int    Cap  int}func main() {    s := make([]int, 0, 2)    sh := (*SliceHeader)(unsafe.Pointer(&s))    fmt.Println(sh)    doSomething(s)}func doSomething(a []int) {    a = append(a, 1)    sh := (*SliceHeader)(unsafe.Pointer(&a))    fmt.Println(sh)}

输出结果:1Sd28资讯网——每日最新资讯28at.com

&{1374389592336 0 2} // main&{1374389592336 1 2} // doSomething

两个切片的 Data 指针地址指向的是同一个底层数组。但由于长度不同,它们在应用的表现上是两个不同的切片。1Sd28资讯网——每日最新资讯28at.com

这也印证了前面问题的结果是输出了空切片,切片长度为 0 的内部原理。1Sd28资讯网——每日最新资讯28at.com

变通方法

这种情况下,建议是修改写法,提高代码易读性。否则后续维护也比较麻烦,不熟悉的同学咋一眼一看很有可能发现不了问题。1Sd28资讯网——每日最新资讯28at.com

但如果你还是希望输出你想要的切片值,可以采取以下变通方法。1Sd28资讯网——每日最新资讯28at.com

改动后的代码:1Sd28资讯网——每日最新资讯28at.com

func main() {    s := make([]int, 0, 2)    doSomething(s)    fmt.Println(s[:1]) // 进行新的切片操作}func doSomething(a []int) {    a = append(a, 1)}

输出结果:1Sd28资讯网——每日最新资讯28at.com

[1]

原因是在进行 s[:1] 切片操作时,本质上是创建了一个新的 SliceHeader,所以可以正常打印和获取预期的元素。1Sd28资讯网——每日最新资讯28at.com

当然,还有一种常见的写法就是切片 append 等变更后一定做一遍再赋值,这样可以规避掉不少使用上的细节坑。1Sd28资讯网——每日最新资讯28at.com

总结

今天这篇文章讨论了一个很常见的 Go 面试问题,内容涉及切片作为函数参数的传递和修改。1Sd28资讯网——每日最新资讯28at.com

重点在于切片作为参数是按值传递的,因此函数内部的修改不会影响外部变量。1Sd28资讯网——每日最新资讯28at.com

如果仍然希望获取可以通过切片操作,重新切分一下新的切片结果集就可以了。1Sd28资讯网——每日最新资讯28at.com

本文链接://www.dmpip.com//www.dmpip.com/showinfo-26-93368-0.htmlGo 面试中的隐藏陷阱:SliceHeader 问题解析

声明:本网页内容旨在传播知识,若有侵权等问题请及时与本网联系,我们将在第一时间删除处理。邮件:2376512515@qq.com

上一篇: C#托管堆遭破坏问题溯源分析

下一篇: 基于 Spring Boot 与 WebSocket 实现实时车位管理与状态更新

标签:
  • 热门焦点
Top
Baidu
map