go slice

// The append built-in function appends elements to the end of a slice. If
// it has sufficient capacity, the destination is resliced to accommodate the
// new elements. If it does not, a new underlying array will be allocated.
// Append returns the updated slice. It is therefore necessary to store the
// result of append, often in the variable holding the slice itself:
//  slice = append(slice, elem1, elem2)
//  slice = append(slice, anotherSlice...)
// As a special case, it is legal to append a string to a byte slice, like this:
//  slice = append([]byte("hello "), "world"...)
func append(slice []Type, elems ...Type) []Type
  • 该函数的作用是追加元素到切片slice的末尾。
  • 如果slice还有剩余的容量,则该切片被允许容纳新的元素;如果没有剩余容量,一个新的底层数组将会创建。
  • 该函数返回更新的切片,因此新的切片必须存储追加结果,通常更新的切片也包含原切片本身。
16291711997043.jpg
package test

import (
    "fmt"
    "testing"
)

func TestSize(t *testing.T) {
    a := []int{1, 2, 3, 4, 5, 6}
    fmt.Printf("aaaa 的地址: %p ,长度 :%d %#v\n", a, len(a), a)
    for i := 0; i < len(a); i++ {
        fmt.Printf("a[%d] 的地址: %p \n", i, &a[i])
    }
  // b 引用a索引为 2的元素
    b := a[2:3]
    fmt.Printf("bbbb 的地址: %p ,长度 :%d %#v\n", b, len(b), b)
    for i := 0; i < 16; i++ {
        b = append(b, 30)
        fmt.Printf("b----b 的地址: %p ,长度 :%d %#v\n", b, len(b), b)
        //fmt.Printf("a----a 的地址: %p ,长度 :%d %#v\n", a, len(a), a)
    }
}
=== RUN   TestSize
aaaa 的地址: 0xc000020420 ,长度 :6 []int{1, 2, 3, 4, 5, 6}
a[0] 的地址: 0xc000020420 
a[1] 的地址: 0xc000020428 
a[2] 的地址: 0xc000020430 // b 的引用地址
a[3] 的地址: 0xc000020438 
a[4] 的地址: 0xc000020440 
a[5] 的地址: 0xc000020448 
b 的地址: 0xc000020430 ,容量: 4,长度 :1 []int{3}
b 的地址: 0xc000020430 ,容量: 4,长度 :2 []int{3, 30}
b 的地址: 0xc000020430 ,容量: 4,长度 :3 []int{3, 30, 30}
b 的地址: 0xc000020430 ,容量: 4,长度 :4 []int{3, 30, 30, 30}

//扩容后,地址改变
b 的地址: 0xc00001a280 ,容量: 8,长度 :5 []int{3, 30, 30, 30, 30}
b 的地址: 0xc00001a280 ,容量: 8,长度 :6 []int{3, 30, 30, 30, 30, 30}
b 的地址: 0xc00001a280 ,容量: 8,长度 :7 []int{3, 30, 30, 30, 30, 30, 30}
b 的地址: 0xc00001a280 ,容量: 8,长度 :8 []int{3, 30, 30, 30, 30, 30, 30, 30}
//扩容后,地址改变
b 的地址: 0xc000116180 ,容量: 16,长度 :9 []int{3, 30, 30, 30, 30, 30, 30, 30, 30}
b 的地址: 0xc000116180 ,容量: 16,长度 :10 []int{3, 30, 30, 30, 30, 30, 30, 30, 30, 30}
b 的地址: 0xc000116180 ,容量: 16,长度 :11 []int{3, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30}
b 的地址: 0xc000116180 ,容量: 16,长度 :12 []int{3, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30}
b 的地址: 0xc000116180 ,容量: 16,长度 :13 []int{3, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30}
b 的地址: 0xc000116180 ,容量: 16,长度 :14 []int{3, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30}
b 的地址: 0xc000116180 ,容量: 16,长度 :15 []int{3, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30}
b 的地址: 0xc000116180 ,容量: 16,长度 :16 []int{3, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30}
//扩容后,地址改变
b 的地址: 0xc00000a700 ,容量: 32,长度 :17 []int{3, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30}
b 的地址: 0xc00000a700 ,容量: 32,长度 :18 []int{3, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30}
b 的地址: 0xc00000a700 ,容量: 32,长度 :19 []int{3, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30}
b 的地址: 0xc00000a700 ,容量: 32,长度 :20 []int{3, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30}
b 的地址: 0xc00000a700 ,容量: 32,长度 :21 []int{3, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30}
b 的地址: 0xc00000a700 ,容量: 32,长度 :22 []int{3, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30}
b 的地址: 0xc00000a700 ,容量: 32,长度 :23 []int{3, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30}
b 的地址: 0xc00000a700 ,容量: 32,长度 :24 []int{3, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30}
b 的地址: 0xc00000a700 ,容量: 32,长度 :25 []int{3, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30}
b 的地址: 0xc00000a700 ,容量: 32,长度 :26 []int{3, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30}
b 的地址: 0xc00000a700 ,容量: 32,长度 :27 []int{3, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30}
b 的地址: 0xc00000a700 ,容量: 32,长度 :28 []int{3, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30}
b 的地址: 0xc00000a700 ,容量: 32,长度 :29 []int{3, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30}
b 的地址: 0xc00000a700 ,容量: 32,长度 :30 []int{3, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30}
b 的地址: 0xc00000a700 ,容量: 32,长度 :31 []int{3, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30}
b 的地址: 0xc00000a700 ,容量: 32,长度 :32 []int{3, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30}

//扩容后,地址改变

b 的地址: 0xc00014e000 ,容量: 64,长度 :33 []int{3, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30}
b 的地址: 0xc00014e000 ,容量: 64,长度 :34 []int{3, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30}
b 的地址: 0xc00014e000 ,容量: 64,长度 :35 []int{3, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30}
b 的地址: 0xc00014e000 ,容量: 64,长度 :36 []int{3, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30}
b 的地址: 0xc00014e000 ,容量: 64,长度 :37 []int{3, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30}
b 的地址: 0xc00014e000 ,容量: 64,长度 :38 []int{3, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30}
b 的地址: 0xc00014e000 ,容量: 64,长度 :39 []int{3, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30}
b 的地址: 0xc00014e000 ,容量: 64,长度 :40 []int{3, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30}
b 的地址: 0xc00014e000 ,容量: 64,长度 :41 []int{3, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30, 30}
--- PASS: TestSize (0.00s)
PASS

Process finished with the exit code 0

从上面程序的输出可以看出

  • 向 b 中追加数据会覆盖 a 中的数据。
  • 当切片b的长度超过数组a固有长度时,b 的地址改变了。
  • 当切片容量被用完时,如果继续追加,则切片扩容后的大小为切片当前大小2 倍。
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • fmt.Printf("%p\n", &xxx)的打印问题 后面的参数必须为 指针类型,否则IDE会有提示,运行后...
    cuishuang阅读 838评论 0 0
  • 原文链接:https://blog.thinkeridea.com/201901/go/slice_de_yi_x...
    戚银阅读 5,469评论 0 0
  • 1. 前言 Slice又称动态数组,依托数组实现,可以方便的进行扩容、传递等,实际使用中比数组更灵活。 正因为灵活...
    淘小铺刀仔阅读 4,026评论 0 0
  • slice的存储结构 slice代表变长的序列,它的底层是数组。一个切片由3部分组成:指针、长度和容量。指针指向底...
    cindywang阅读 10,785评论 0 0
  • 切片(slice)是对数组的一个连续片段的引用,这个连续片段可以是整个数组,也可以是由起止索引标识的子集元素。需要...
    JunChow520阅读 2,890评论 0 1