开门见山
今天发现一个十分有趣的case,如下:
package main
import "fmt"
func main() {
n1 := make ([] int, 0,5)
n2 := n1[:2]
fmt.Println(n1)
fmt.Println(n2)
// 思考 n1和n2打印出的指针地址是否相同?
fmt.Printf("address of n1:%p\n",n1)
fmt.Printf("address of n2:%p\n",n2)
}
n1不是指针类型,是make的一个[]int slice的引用类型,可能多数人会有这样的思考: 两个对象的指针地址相同,那么这两个对象存储的内容是相同的,即使slice的底层数组结构SliceHeader的Data字段域(数组),在未超出n1的容量5之前,和n2用的是一个Data字段域,但是因为sliceHeader里第二个字段域是Len,第三个是Cap,n1和n2的SliceHeader肯定不是同一个,那么用%p打印出的指针地址肯定是不同的指针地址,我开始也是这样的想法,然后结果让我大吃一惊, 如下:
[]
[0 0]
address of n1:0xc00007e030
address of n2:0xc00007e030
什么? 打印出的指针地址相同。我瞬间就不淡定了 这似乎颠覆了我的一个固定观念–两个对象的指针地址相同,那么这两个对象存储的内容是相同的
猜想
带着自己的疑惑 我首先想到的是利用反射,把slice转成SliceHeader来一探究竟 看