Golang 切片详解一 内存布局

 不说废话,直接上货,能懂则懂,不能懂则略过

package main

import (
	"fmt"
	"unsafe"
)

func main() {

	fmt.Println()

	// 切片
	// 创建一个切片给变量a,这个切片的长度为0,其底层数组长度为8(常称之为容量)
	// 切片a的数据结构 struct{ptr: 0x0, len: 0, cap: 8} 占用24字节(72位)
	// ptr:占8字节 是一个指针 存储的是底层数组的内存地址
	// len:占8字节 是切片的长度
	// cap: 占8字节 是底层数组的长度
	a := make([]int8, 0, 10)
	b := append(a, 2)
	_ = append(a, 3)
	_ = append(a, []int8{4, 5}...)
	_ = append(a, []int8{4, 5, 6}...)
	_ = append(a, []int8{4, 5, 6, 7}...)

	// 切片a的数据结构
	ptra := (*uint64)(unsafe.Pointer(&a))                                  // 占8字节 指向的是底层数组
	ptral := *(*int64)(unsafe.Pointer(uintptr(unsafe.Pointer(ptra)) + 8))  // 占8字节 切片的长度
	ptrac := *(*int64)(unsafe.Pointer(uintptr(unsafe.Pointer(ptra)) + 16)) // 占8字节 切片的容量

	fmt.Printf("切片a的地址:%#x 底层数组:0x%x 长度:%d 容量:%d 数据:%v\n", ptra, *ptra, ptral, ptrac, a)
	// 切片a的地址:0xc00011c460 底层数组:0xc000126090 长度:0 容量:10 数据:[]

	// 切片b的数据结构
	ptrb := (*uint64)(unsafe.Pointer(&b))                                  // 占8字节 指向的是底层数组
	ptrbl := *(*int64)(unsafe.Pointer(uintptr(unsafe.Pointer(ptrb)) + 8))  // 占8字节 切片的长度
	ptrbc := *(*int64)(unsafe.Pointer(uintptr(unsafe.Pointer(ptrb)) + 16)) // 占8字节 切片的容量

	fmt.Printf("切片b的地址:%#x 底层数组:0x%x 长度:%d 容量:%d 数据:%v\n", ptrb, *ptrb, ptrbl, ptrbc, b)
	// 切片b的地址:0xc00011c480 底层数组:0xc000126090 长度:1 容量:10 数据:[4]

	p := uintptr(*ptra)
	for i := int64(0); i < ptrac; i++ {
		p := (*int8)(unsafe.Pointer(p + uintptr(i)))
		fmt.Printf("addr:%p key: %d val: %v\n", p, i, *p)
	}
	// 底层数组
	// addr:0xc000126090 key: 0 val: 4
	// addr:0xc000126091 key: 1 val: 5
	// addr:0xc000126092 key: 2 val: 6
	// addr:0xc000126093 key: 3 val: 7
	// addr:0xc000126094 key: 4 val: 0
	// addr:0xc000126095 key: 5 val: 0
	// addr:0xc000126096 key: 6 val: 0
	// addr:0xc000126097 key: 7 val: 0
	// addr:0xc000126098 key: 8 val: 0
	// addr:0xc000126099 key: 9 val: 0

	fmt.Printf("\n")
}

 

Golang 切片详解一 内存布局_第1张图片

 

Golang 切片详解一 内存布局_第2张图片

 

Golang 切片详解一 内存布局_第3张图片

 

Golang 切片详解一 内存布局_第4张图片

 

 

 Golang 切片详解一 内存布局_第5张图片

 

综上述:

  1. 切片是一个数据结构体,拥有3段数据,每段占8字节,共占24字节。
  2. 第一段,存储切片的底层数组所在的内存地址,即指针
  3. 第二段,存储着切片的长度,即切片本身的长度
  4. 第三段,存储着切片的容量,即底层数组的长度

你可能感兴趣的:(Golang)