切片slice:切片是对数组的抽象。切片在内存中占24个字节
runtime.h
struct Slice{ // must not move anything
byte* array; // actual data
uintgo len; // number of elements
uintgo cap; // allocated number of elements
};
切片包含长度、容量、以及一个指向首元素的指针
• 引⽤类型。但⾃⾝是结构体,值拷⻉传递。
• 属性 len 表⽰可⽤元素数量,读写操作不能超过该限制。
• 属性 cap 表⽰最⼤扩张容量,不能超出数组限制。
• 如果 slice == nil,那么 len、 cap 结果都等于 0。
• 作为变长数组的替代方案,可以关联底层数组的局部或全部
•可以直接创建或从底层数组获取生成
• 使用len()获取元素个数,cap()获取容量
• 一般使用make()创建
•如果多个slice指向相同底层数组,其中一个值的改变会影响全部
•在通过下标访问元素时下标不能超过len大小,如同数组的下标不能超出len范围一样。
make([]T, len, cap)
其中cap可以省略,则和len的值相同
len表示存放的元素个数,cap表示容量
1、初始化的几种方式
//第一种方式创建切片
var slice []int
fmt.Println(len(slice)) //0
//第二种方式
sl1 := []int{0, 1, 2, 3, 8: 100} // 通过初始化表达式构造,可使⽤索引号。
fmt.Println(sl1, len(sl1), cap(sl1)) //[0 1 2 3 0 0 0 0 100] 9 9
sl2 := make([]int, 10) // 使⽤ make 创建,省略 cap,相当于 cap = len。
fmt.Println(sl2) //[0 0 0 0 0 0 0 0 0 0]
//第三种方式
num := []int{10, 20, 30, 40, 50}
fmt.Println(num) //[10 20 30 40 50]
2、append
向 slice 尾部添加数据,返回新的 slice 对象
切片可以通过内置函数append(slice []Type,elems …Type)追加元素,elems可以是一排type类型的数据,也可以是slice,因为追加的一个一个的元素,因此如果将一个slice追加到另一个slice中需要带上”…”,这样才能表示是将slice中的元素依次追加到另一个slice中。append追加元素超出实际容量会执行扩容,会扩展为slice原先容量的2倍
(1)//将一个slice追加到另一个slice中需要带上”…”,这样表示是将slice中的元素依次追加到另一个slice中
例:
veggies := []string{"potatoes", "tomatoes", "brinjal"}
fruits := []string{"oranges", "apples"}
food := append(veggies, fruits...) //veggies+fruits
fmt.Println("food:", food)
(2)切片元素删除,可以使用append来实现
s = append(s[:i], s[i+1:]…)
首先s[:i]相当于slice截取,也就是说s[:i]本身就是一个slice。然后s[i+1:]…相当于变长参数。使用append的特性(向 slice 尾部添加数据,返回新的 slice 对象)来实现删除的功能。可以单个也可以删除多个。
例:
//删除scile中元素,删除下标为2的元素
test := []int{10, 20, 30, 40, 50, 100}
test = append(test[:2], test[3:]...)
fmt.Println(test) //[10 20 40 50 100]
3、copy
函数 copy 在两个 slice 间复制数据,复制⻓度以 len ⼩的为准。两个 slice 可指向同⼀底层数组,允许元素区间重叠。
例:
//copy
countries := []string{"USA", "Singapore", "Germany", "India", "Australia"}
neededCountries := countries[:len(countries)-2]
countriesCpy := make([]string, len(neededCountries))
copy(countriesCpy, neededCountries)
fmt.Println(countriesCpy)
fmt.Println(len(countriesCpy), cap(countriesCpy))
data := [...]int{0, 1, 2, 3, 4, 5, 6, 7, 8, 9}
s := data[8:]
s2 := data[:5]
copy(s2, s) // dst:s2, src:s
fmt.Println(s2) //[8 9 2 3 4]
fmt.Println(data) //[8 9 2 3 4 5 6 7 8 9]
应及时将所需数据 copy 到较⼩的 slice,以便释放内存。
4、如果两个切片共享同一块地址空间,使用append删除其中一个的参数,另一个切片虽然会被影响,但len不变,会自动在最后补上len长度的参数
demo
func main() {
a1 := []int{1, 2, 3, 4, 5}
a2 := a1
a1 = append(a1[:2], a1[3:]...)
fmt.Println(a1) //[1 2 4 5]
fmt.Println(a2) //[1 2 4 5 5]
}
下面是整个例子
package main
import (
"fmt"
"unsafe"
)
func main() {
//第一种方式创建切片
var slice []int
fmt.Println(len(slice)) //0
//第二种方式
sl1 := []int{0, 1, 2, 3, 8: 100} // 通过初始化表达式构造,可使⽤索引号。
fmt.Println(sl1, len(sl1), cap(sl1)) //[0 1 2 3 0 0 0 0 100] 9 9
sl2 := make([]int, 10) // 使⽤ make 创建,省略 cap,相当于 cap = len。
fmt.Println(sl2) //[0 0 0 0 0 0 0 0 0 0]
//第三种方式
num := []int{10, 20, 30, 40, 50}
fmt.Println(num) //[10 20 30 40 50]
fmt.Printf("type is %T\tsize is %d\n", sl2, unsafe.Sizeof(sl2))
var b []int
b = num[1:4] //左闭右开从num下标1到3
var b1 []int
b1 = num[:] //num的全部
fmt.Println(b)
fmt.Println(b1)
//切片为数组的引用
darr := [...]int{57, 89, 90, 82, 100, 78, 67, 69, 59}
dslice := darr[2:5]
fmt.Println("array before", darr)
for i := range dslice {
dslice[i]++
}
fmt.Println("array after", darr)
//切片指向数组,容量cap会从strartIndex取到数组结束,长度是指定截取长度
fruitarray := [...]string{"apple", "orange", "grape", "mango", "water melon",
"pine apple", "chikoo"}
fruitslice := fruitarray[0:3]
//长度2容量6
fmt.Printf("length of slice %d capacity %d", len(fruitslice), cap(fruitslice))
//append追加元素超出实际容量会执行扩容,会扩展为原先容量的2倍
slice1 := make([]int, 5, 10)
fmt.Println(slice1)
slice3 := append(slice1, 1, 2, 3, 4, 5)
fmt.Println(slice3)
//执行append会会返回一个新的数值可以用原切片接收也可以使用别的
var slice4 []int //空的切片初始为nil
fmt.Println(slice4)
if slice4 == nil {
slice4 = append(slice4, 1, 2, 3, 4)
fmt.Println(slice4)
}
//将一个slice追加到另一个slice中需要带上"…",这样表示是将slice中的元素依次追加到另一个slice中
veggies := []string{"potatoes", "tomatoes", "brinjal"}
fruits := []string{"oranges", "apples"}
food := append(veggies, fruits...) //veggies+fruits
fmt.Println("food:", food)
//二维切片,每一行元素的个数可以不一致
pls := [][]string{
{"C", "C++"},
{"JavaScript"},
{"Go", "Rust"},
}
fmt.Println(len(pls), cap(pls)) //长度为3容量为3
for _, v1 := range pls {
for _, v2 := range v1 {
fmt.Printf("%s ", v2)
}
fmt.Printf("\n")
}
//copy
countries := []string{"USA", "Singapore", "Germany", "India", "Australia"}
neededCountries := countries[:len(countries)-2]
countriesCpy := make([]string, len(neededCountries))
copy(countriesCpy, neededCountries)
fmt.Println(countriesCpy)
fmt.Println(len(countriesCpy), cap(countriesCpy))
data := [...]int{0, 1, 2, 3, 4, 5, 6, 7, 8, 9}
s := data[8:]
s2 := data[:5]
copy(s2, s) // dst:s2, src:s
fmt.Println(s2) //[8 9 2 3 4]
fmt.Println(data) //[8 9 2 3 4 5 6 7 8 9]
//删除scile中元素,删除下标为2的元素
test := []int{10, 20, 30, 40, 50, 100}
test = append(test[:2], test[3:]...)
fmt.Println(test) //[10 20 40 50 100]
}