golang-数组和切片的区别

tydldd 2019-09-08

数组:

语法:

//value := [数据长度]类型 {}

arr := [1]string{"1"}// 声明并且赋值

arr = [1]string{}// 声明未赋值
arr[0]="1"

数组类型的值(以下简称数组)的长度是固定的数组的长度在声明它的时候就必须给定,并且在之后不会再改变。可以说,数组的长度是其类型的一部分(数组的容量永远等于其长度,都是不可变的)

切片:

语法:

//value := []类型 {}

// 创建长度容量都为0的切片
value :=[]string{}// []string{"1"} 长度容量为1的切片
value :=append(value,"1")

//数据转切片
arr := [1]string{"1"}
slice :=arr[:]

// 创建长度容量为1的切片
slice = make([]string, 1)
// 创建长度为1,容量为2的切片
slice = make([]string, 1,2)

切片类型的值是可变长的。而切片的类型字面量中只有其元素的类型,而没有其长度。切片的长度可以自动地随着其中元素数量的增长而增长,但不会随着元素数量的减少而减少。

在每一个切片的底层数据结构中,会包含一个数组,可以被叫做底层数据,而切片就是对底层数组的引用,故而切片类型属于引用类型

切片理解

底层数据类似一个窗户,窗户的宽度就类似与底层数据的长度,而切片就是窗口(该窗口只能往一个方向移动),可以通过该窗口看到一个数组,但不一定能看到该数组中所有的元素,有时候只能看待连续的一部分元素

代码:

/*
// 切片的结构体
type slice struct {
    array unsafe.Pointer // 底层数据的指针
    len   int // 切的长度
    cap   int // 截取底层数据的容量
}

*/

arr := [8]int{1, 2, 3, 4, 5, 6, 7, 8}//定义一个数组,作为底层数组
/*
    进行切片操作,将 索引为3开始到后面的元素截取出来
    sli := slice{
        array: unsafe.Pointer(&arr),
        len:   6 - 3,
        cap:   cap(arr) - 3,
    }
    sli容量 表示arr中的 { 4, 5, 6, 7, 8}这部分元素
    sli:表示arr中的 {4,5,6},因为长度是3 就取容量表示的三个元素
*/

sli := arr[3:6]
fmt.Printf("slice: %d\n", sli)//slice: [4 5 6

/*
    再进行切片操作的时候,
    sli[:cap(sli)]
    slitemp := slice{
        array: unsafe.Pointer(&sli),
        len:   cap(sli) - 0, // 5 -0
        cap:   cap(arr) - 0,// 5-0
    }
    slitemp: 表示arr中的 { 4, 5, 6, 7, 8}这部分元素

*/
fmt.Printf("slice: %d\n", sli[:cap(sli)])//slice: [4 5 6 7 8]

共同点:

都属于集合类的类型,它们的值也都可以用来存储某一种类型的值(或者说元
素)。

扩展:

一个切片无法容纳更多的元素,Go 语言就会想办法扩容。但它并不会改变原来的切片,而
是会生成一个容量更大的切片,然后将把原有的元素和新元素一并拷贝到新切片中。
在一般的情况下,你可以简单地认为新切片的容量(以下简称新容量)将会是原切片容量(以下
简称原容量)的 2 倍。
但是,当原切片的长度(以下简称原长度)大于或等于1024时,Go 语言将会以原容量的1.25
倍作为新容量的基准(以下新容量基准)。
新容量基准会被调整(不断地与1.25相乘),直到结果不小于原长度与要追加的元素数量之和
(以下简称新长度)。

只要新长度不会超过切片的原容量,那么使用append函数对其追加元素的时候就
不会引起扩容。这只会使紧邻切片窗口右边的(底层数组中的)元素被新的元素替换掉。