Go 語言 切片的使用(增刪改查)
阿新 • • 發佈:2022-03-25
Go 語言 切片的使用(增刪改查)
引言
Golang 的陣列是固定長度,可以容納相同資料型別的元素的集合。
但是當長度固定了,在使用的時候肯定是會帶來一些限制,比如說:申請的長度太大會浪費記憶體,太小又不夠用。
鑑於上述原因,我們有了 go 語言的切片,可以把切片理解為,可變長度的陣列,其實它底層就是使用陣列實現的,增加了自動擴容功能。
切片(Slice)是一個擁有相同型別元素的可變長度的序列。
一、切片的基礎語法
二、切片的初始化
三、切片的遍歷
四、切片元素的新增和刪除copy
一、切片的基礎語法
1. 語法
宣告一個切片和宣告一個數組類似,只要不新增長度即可
var identifier []type
切片是引用型別,可以使用make函式
來建立切片:
var slice1 []type =make([]type, len) 也可以簡寫為 slice1 := make([]type, len)
也可以指定容量,其中capacity為可選引數
make([]T,length,capacity)
這裡len是陣列的長度並且也是切片的初始長度
2. 示例
切片在沒有賦值之前是空值
宣告變數的方式建立切片
package main import "fmt" func main() { var names []string var numbers []int fmt.Printf("names: %v\n", names) fmt.Printf("numbers: %v\n", numbers) fmt.Println(names == nil) fmt.Println(numbers == nil) } //輸出結果如下 names: [] numbers: [] true true
使用make函式
建立切片
package main import "fmt" func main() { var a = make([]int, 2) var b = make([]int, 3) fmt.Printf("a: %v\n", a) fmt.Printf("b: %v\n", b) } //輸出結果如下 a: [0 0] b: [0 0 0]
3. 切片的長度和容量
切片擁有自己的長度和容量,我們可以通過使用內建的len()函式
求出長度,使用內建的cap()函式
求出切片的內容。
package main import "fmt" func main() { var names = []string{"hello", "world"} var num = []int{1, 2, 3} fmt.Printf("len: %d cap: %d\n", len(names), cap(names)) fmt.Printf("len: %d cap: %d\n", len(num), cap(num)) fmt.Printf("num[2]: %v\n", num[2])//根據下標索引來獲取切片的元素 fmt.Println("-----------") var a = make([]string, 2, 3) fmt.Printf("len: %d cap: %d", len(a), cap(a)) } //輸出結果如下 len: 2 cap: 2 len: 3 cap: 3 num[2]: 3 ----------- len: 2 cap: 3
二、切片的初始化
切片的初始化方式有很多種,可以直接初始化,也可以使用陣列初始化。
1. 直接初始化
package main import "fmt" func main() { a := []int{1, 2, 3} fmt.Printf("a: %v\n", a) } //輸出結果如下 a: [1 2 3]
2. 使用陣列初始化
package main import "fmt" func main() { a := [...]int{1, 2, 3} b := a[:] fmt.Printf("b: %v\n", b) } //輸出結果 b: [1 2 3]
3. 使用陣列的部分元素初始化(切片表示式)
- 切片的底層就是一個
陣列
,所以我們可以基於陣列通過切片表示式得到切片
。 - 切片表示式中的
low
和high
表示一個索引範圍(包左不包右)
,得到切片的長度=high-low
,容量等於得到的切片的底層陣列的容量。
package main import "fmt" func main() { a := [...]int{1, 2, 3, 4, 5, 6, 7, 8} b := a[2:5] //下標2到5,左閉右開,不包括5 fmt.Printf("b: %v\n", b) c := a[2:] //下標2後面的所有 fmt.Printf("c: %v\n", c) d := a[:3] //下標3之前的,不包括3 fmt.Printf("d: %v\n", d) e := a[:] //取所有值 fmt.Printf("e: %v\n", e) } //輸出結果如下 b: [3 4 5] c: [3 4 5 6 7 8] d: [1 2 3] e: [1 2 3 4 5 6 7 8]
4. 空(nil)切片
一個切片在未初始化之前預設為nil,長度為0,容量為0
package main import "fmt" func main() { var a []int fmt.Println(a == nil) fmt.Printf("len: %d,cap: %d\n", len(a), cap(a)) } //輸出結果如下 true len: 0,cap: 0
三、切片的遍歷
切片的遍歷和陣列的遍歷非常型別,可以使用for
迴圈索引遍歷,或者for range
迴圈
1. for 迴圈遍歷
package main import "fmt" func main() { s1 := []int{1, 2, 3, 4, 5, 6} for i := 0; i < len(s1); i++ { fmt.Printf("s1[%d]: %v\n", i, s1[i]) } } //輸出結果如下 s1[0]: 1 s1[1]: 2 s1[2]: 3 s1[3]: 4 s1[4]: 5 s1[5]: 6
2. for range遍歷
package main import "fmt" func main() { s1 := []int{1, 2, 3, 4, 5, 6} for i, v := range s1 { fmt.Printf("i: %v,v: %v\n", i, v) } } //輸出結果如下,i是索引,v是值 i: 0,v: 1 i: 1,v: 2 i: 2,v: 3 i: 3,v: 4 i: 4,v: 5 i: 5,v: 6
四、切片元素的新增和刪除copy
- 切片是一個動態陣列,可以使用
append()函式
新增元素 - go 語言中並沒有刪除切片元素的專用方法,我們可以使用切片本身的特性來刪除元素。
- 由於切片是引用型別,通過賦值的方式,會修改原有內容,go 提供了
copy()函式
來拷貝切片
1. 新增元素
package main import "fmt" func main() { a := []int{} a = append(a, 1) a = append(a, 2) a = append(a, 3, 4, 5) //新增多個元素 fmt.Printf("a: %v\n", a) fmt.Println("---------------------") a1 := []int{3, 4, 5} a2 := []int{1, 2} a2 = append(a2, a1...) //新增另外一個切片進去 fmt.Printf("a2: %v\n", a2) } //輸出結果如下 a: [1 2 3 4 5] --------------------- a2: [1 2 3 4 5]
2. 刪除元素
package main import "fmt" func main() { var s1 = []int{1, 2, 3, 4} fmt.Println("---刪除之前---") fmt.Printf("s1: %v\n", s1) //刪除3這個元素,它的下標索引為2 s1 = append(s1[:2], s1[3:]...) fmt.Println("---刪除之後---") fmt.Printf("s1: %v\n", s1) } //執行結果如下 ---刪除之前--- s1: [1 2 3 4] ---刪除之後--- s1: [1 2 4]
刪除元素的公式
要從切片a中刪除索引為index的元素,操作方法如下 a = append(a[:index], a[index+1:]...)
3. 修改切片元素
package main import "fmt" func main() { var s1 = []int{1, 2, 3, 4, 5} s1[1] = 100 //索引1的值改為100 fmt.Printf("s1: %v\n", s1) } //輸出結果如下 s1: [1 100 3 4 5]
4. 查詢切片元素
package main import "fmt" func main() { var s1 = []int{1, 2, 3, 4, 5} var key = 2 //查詢2所在的位置 for i, v := range s1 { if v == key { fmt.Printf("s1: %v\n", s1) fmt.Printf("索引為: %v\n", i) } } } //輸出結果如下 s1: [1 2 3 4 5] 索引為: 1
5. 拷貝切片
package main import "fmt" func main() { var s1 = []int{1, 2, 3, 4, 5} var s2 = s1 s2[0] = 100 fmt.Printf("s1: %v\n", s1) fmt.Printf("s2: %v\n", s2) } //輸出結果如下 s1: [100 2 3 4 5] s2: [100 2 3 4 5]
可以看到當s2的值改變的時候,s1的值也改變了,因為s2複製了s1的是他的記憶體地址,所以相關聯都會改變
- 使用
copy
方法修改則不會對源切片的值有影響
package main import "fmt" func main() { var s1 = []int{1, 2, 3, 4, 5} var s2 = make([]int, 4) //需要make一個切片的型別,指定有5個元素 copy(s2, s1) //指定複製的切片 s2[0] = 100 fmt.Printf("s1: %v\n", s1) fmt.Printf("s2: %v\n", s2) } //輸出結果 s1: [1 2 3 4 5] s2: [100 2 3 4 5]