【golang】切片Slice

切片Slice

需要说明,slice 并不是数组或数组指针。它通过内部指针和相关属性引用数组片段,以实现变长方案。

1. 切片:切片是数组的一个引用,因此切片是引用类型。但自身是结构体,值拷贝传递。
2. 切片的长度可以改变,因此,切片是一个可变的数组。
3. 切片遍历方式和数组一样,可以用len()求长度。表示可用元素数量,读写操作不能超过该限制。 
4. cap可以求出slice最大扩张容量,不能超出数组限制。0 <= len(slice) <= len(array),其中array是slice引用的数组。
5. 切片的定义:var 变量名 []类型,比如 var str []string  var arr []int。
6. 如果 slice == nil,那么 len、cap 结果都等于 0。

1.1.1、创建切片的各种方式

package main
import "fmt"
func main(){
	//1.声明切片
	var s1 []int
	if s1 == nil{
		fmt.Println("是空")
	}else{
		fmt.Println("不是空")
	}
	//2.:=
	s2 := []int{}
	//3.make()
	var s3 []int = make([]int,0)
	fmt.Println(s1,s2,s3)
	//4.初始化赋值
	var s4 []int = make([]int,0,0)
	fmt.Println(s4)
	s5 := []int{1,2,3}
	fmt.Println(s5)
	//5.从数组切片
	arr := [5]int{1,2,3,4,5}
	var s6 []int
	//前包后不包
	s6 = arr[1:4]
	fmt.Println(s6)
}

1.1.2、切片初始化

全局:
var arr=[...]int{0,1,2,3,4,5,6,7,8,9}
var slice0 []int = arr[start:end]
var slice1 []int = arr[:end]
var slice2 []int = arr[start:]
var slice3 []int = arr[:]
var slice4 []int = arr[:len(arr)-1] //去掉切片的最后一个元素
局部:
arr2 := [...]int{9,8,7,6,5,4,3,2,1,0}
slice5 := arr[start:end]
slice6 := arr[:end]
slice7 := arr[start:]
slice8 := arr[:]
slice9 := arr[:len(arr)-1]//去掉切片的最后一个元素
操作含义
s[n]切片s中索引位置为n的项
s[:]从切片s的索引位置0到len(s)-1处所获得的切片
s[low:]从切片s的索引位置low到len(s)-1处所获得的切片
s[:high]从切片s的索引位置0到high处所获得的切片,len=high
s[low:high]从切片s的索引位置low到high处所获得的切片,len=high-low
s[low:high:max]从切片s的索引位置low到high处所获得的切片,len=high-low,cap=max-low
len(s)切片s的长度,总是<=cap(s)
cap(s)切片s的容量,总是>=len(s)

代码:

package main
import(
	"fmt"
)
var arr = [...]int{0,1,2,3,4,5,6,7,8,9}
var slice0 []int = arr[2:8]
var slice1 []int = arr[0:6] //可以简写为 var slice []int = arr[:end]
var slice2 []int = arr[5:10] //可以简写为var slice[]int = arr[start:]
var slice3 []int = arr[0:len(arr)] // var slice []int = arr[:]
var slice4 = arr[:len(arr)-1] //去掉切片的最后一个元素
func main(){
	fmt.Printf("全局变量:arr %v\n",arr)
	fmt.Printf("全局变量:slice0 %v\n",slice0)
	fmt.Printf("全局变量:slice1 %v\n",slice1)
	fmt.Printf("全局变量:slice2 %v\n",slice2)
	fmt.Printf("全局变量:slice3 %v\n",slice3)
	fmt.Printf("全局变量:slice4 %v\n",slice4)
	fmt.Printf("--------------------\n")
	arr2 := [...]int{9,8,7,6,5,4,3,2,1,0}
	slice5 := arr[2:8]
	slice6 := arr[0:6] //可以简写为slice := arr[:end]
	slice7 := arr[5:10] //可以简写为slice := arr[start:]
	slice8 := arr[0:len(arr)] //slice := arr[:]
	slice9 := arr[:len(arr)-1] //去掉切片的最后一个元素
	fmt.Printf("局部变量:arr2 %v\n",arr2)
	fmt.Printf("局部变量:slice5 %v\n",slice5)
	fmt.Printf("局部变量:slice6 %v\n",slice6)
	fmt.Printf("局部变量:slice7 %v\n",slice7)
	fmt.Printf("局部变量:slice8 %v\n",slice8)
	fmt.Printf("局部变量:slice9 %v\n",slice9)
}

输出结果:

全局变量:arr[0 1 2 3 4 5 6 7 8 9]
全局变量:slice0[2 3 4 5 6 7]
全局变量:slice1[0 1 2 3 4 5]
全局变量:slice2[5 6 7 8 9]
全局变量:slice3[0 1 2 3 4 5 6 7 8 9]
全局变量:slice4[0 1 2 3 4 5 6 7 8]
------------------------------------
局部变量:arr2[9 8 7 6 5 4 3 2 1 0]
局部变量:slice5[2 3 4 5 6 7]
局部变量:slice6[0 1 2 3 4 5]
局部变量:slice7[5 6 7 8 9]
局部变量:slice8[0 1 2 3 4 5 6 7 8 9]
局部变量:slice9[0 1 2 3 4 5 6 7 8]

1.1.3、通过make来创建切片

var slice []type = make([]type,len)
slice := make([]type,len)
slice := make([]type,len,cap)

在这里插入图片描述
代码:

package main

import (
    "fmt"
)

var slice0 []int = make([]int, 10)
var slice1 = make([]int, 10)
var slice2 = make([]int, 10, 10)

func main() {
    fmt.Printf("make全局slice0 :%v\n", slice0)
    fmt.Printf("make全局slice1 :%v\n", slice1)
    fmt.Printf("make全局slice2 :%v\n", slice2)
    fmt.Println("--------------------------------------")
    slice3 := make([]int, 10)
    slice4 := make([]int, 10)
    slice5 := make([]int, 10, 10)
    fmt.Printf("make局部slice3 :%v\n", slice3)
    fmt.Printf("make局部slice4 :%v\n", slice4)
    fmt.Printf("make局部slice5 :%v\n", slice5)
}

输出结果:

	make全局slice0 :[0 0 0 0 0 0 0 0 0 0]
    make全局slice1 :[0 0 0 0 0 0 0 0 0 0]
    make全局slice2 :[0 0 0 0 0 0 0 0 0 0]
    --------------------------------------
    make局部slice3 :[0 0 0 0 0 0 0 0 0 0]
    make局部slice4 :[0 0 0 0 0 0 0 0 0 0]
    make局部slice5 :[0 0 0 0 0 0 0 0 0 0]

切片的内存布局
在这里插入图片描述
读写操作实际目标是底层数组,只需注意索引号的差别。

package main

import (
    "fmt"
)

func main() {
    data := [...]int{0, 1, 2, 3, 4, 5}

    s := data[2:4]
    s[0] += 100
    s[1] += 200

    fmt.Println(s)
    fmt.Println(data)
}

输出:

    [102 203]
    [0 1 102 203 4 5]

可直接创建slice对象,自动分配底层数组。

package main

import "fmt"

func main() {
    s1 := []int{0, 1, 2, 3, 8: 100} // 通过初始化表达式构造,可使用索引号。
    fmt.Println(s1, len(s1), cap(s1))

    s2 := make([]int, 6, 8) // 使用 make 创建,指定 len 和 cap 值。
    fmt.Println(s2, len(s2), cap(s2))

    s3 := make([]int, 6) // 省略 cap,相当于 cap = len。
    fmt.Println(s3, len(s3), cap(s3))
}

输出结果:

    [0 1 2 3 0 0 0 0 100] 9 9
    [0 0 0 0 0 0] 6 8
    [0 0 0 0 0 0] 6 6

使用make动态创建slice,避免了数组必须用常量做长度的麻烦。还可用指针直接访问底层数组,退化成普通数组操作。

package main

import "fmt"

func main() {
    s := []int{0, 1, 2, 3}
    p := &s[2] // *int, 获取底层数组元素指针。
    *p += 100

    fmt.Println(s)
}

输出结果:

    [0 1 102 3]

至于[][]T,是指元素类型为[]T.

package main

import (
    "fmt"
)

func main() {
    data := [][]int{
        []int{1, 2, 3},
        []int{100, 200},
        []int{11, 22, 33, 44},
    }
    fmt.Println(data)
}

输出结果:

    [[1 2 3] [100 200] [11 22 33 44]]
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值