首頁 > 軟體

Go語言實現切片增刪改查的範例程式碼

2022-04-21 19:00:15

引言

Golang 的陣列是固定長度,可以容納相同資料型別的元素的集合。

但是當長度固定了,在使用的時候肯定是會帶來一些限制,比如說:申請的長度太大會浪費記憶體,太小又不夠用。

鑑於上述原因,我們有了 go 語言的切片,可以把切片理解為,可變長度的陣列,其實它底層就是使用陣列實現的,增加了自動擴容功能。

切片(Slice)是一個擁有相同型別元素的可變長度的序列。

一、切片的基礎語法

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: %vn", names)
    fmt.Printf("numbers: %vn", 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: %vn", a)
    fmt.Printf("b: %vn", 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: %dn", len(names), cap(names))
    fmt.Printf("len: %d cap: %dn", len(num), cap(num))
    fmt.Printf("num[2]: %vn", 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: %vn", a)
}

輸出結果如下

a: [1 2 3]

2. 使用陣列初始化

package main
 
import "fmt"
 
func main() {
    a := [...]int{1, 2, 3}
    b := a[:]
    fmt.Printf("b: %vn", 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: %vn", b)
    c := a[2:] //下標2後面的所有
    fmt.Printf("c: %vn", c)
    d := a[:3] //下標3之前的,不包括3
    fmt.Printf("d: %vn", d)
    e := a[:] //取所有值
    fmt.Printf("e: %vn", 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: %dn", 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]: %vn", 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: %vn", 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: %vn", a)
    fmt.Println("---------------------")
    a1 := []int{3, 4, 5}
    a2 := []int{1, 2}
    a2 = append(a2, a1...) //新增另外一個切片進去
    fmt.Printf("a2: %vn", 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: %vn", s1)
 
    //刪除3這個元素,它的下標索引為2
    s1 = append(s1[:2], s1[3:]...)
    fmt.Println("---刪除之後---")
    fmt.Printf("s1: %vn", 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: %vn", 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: %vn", s1)
            fmt.Printf("索引為: %vn", 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: %vn", s1)
    fmt.Printf("s2: %vn", 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: %vn", s1)
    fmt.Printf("s2: %vn", s2)
}

輸出結果

s1: [1 2 3 4 5]
s2: [100 2 3 4 5]

到此這篇關於Go語言實現切片增刪改查的範例程式碼的文章就介紹到這了,更多相關Go語言切片增刪改查內容請搜尋it145.com以前的文章或繼續瀏覽下面的相關文章希望大家以後多多支援it145.com!


IT145.com E-mail:sddin#qq.com