Go to Learn Go之切片

简介: Go to Learn Go之切片

概述

在上一节的内容中,我们介绍了Go的数组,包括:声明数组、初始化数组、访问数组元素等。在本节中,我们将介绍Go的切片。在Go语言中,数组的长度是固定的,不能改变,这在某些场景下使用不太方便。切片(slice)是一种动态数组,它提供了更为灵活和便捷的方式来操作数组。切片是对数组的抽象,它包含了指向数组元素的指针、切片的长度和容量。

声明切片

切片的声明使用[]操作符,语法如下:

 var sliceName []Type

其中,sliceName表示切片的名称,Type表示切片中元素的类型。与数组不同,声明切片时,不需要指定长度。

在下面的示例代码中,我们声明了一个名为numbers的整数类型切片,并初始为空切片。

var numbers []int


初始化切片

可以使用数组来初始化切片,语法如下:

 var sliceName []Type = []Type{value1, value2, ..., valueN}

其中,Type表示切片中元素的类型,value1、value2、...、valueN是切片中元素的初始值。

在下面的示例代码中,我们创建了一个包含3个整数的切片,其初始值为1、2、3。还创建了一个包含2个字符串的切片,其初始值为“Hello”、“World”。

package main
import "fmt"
func main() {
    var numbers []int = []int{1, 2, 3}
    text := []string{"Hello", "World"}
    // 输出:[1 2 3]
    fmt.Println(numbers)
    // 输出:[Hello World]
    fmt.Println(text)
}


我们还可以使用内置函数make()来初始化切片,传入类型、数量、容量(可忽略)即可,元素的初始值为类型的默认值。

package main
import "fmt"
func main() {
    var numbers []int = make([]int, 3)
    text := make([]string, 2)
    // 输出:[0 0 0]
    fmt.Println(numbers)
    // 输出:[ ]
    fmt.Println(text)
}


与数组相比,切片的长度是不固定的,可以追加元素。在追加时,可能使切片的容量增大。切片的长度可以由 len()函数获取,容量可以由cap()函数获取。

package main
import "fmt"
func main() {
    var numbers []int = make([]int, 3, 10)
   
    // 输出:3 10
    fmt.Println(len(numbers), cap(numbers))
}



如果切片声明后,没有初始化,则为空切片。空切片默认为nil,其长度和容量均为0。

package main
import "fmt"
func main() {
    var numbers []int
   
    // 输出:0 0
    fmt.Println(len(numbers), cap(numbers))
    if numbers == nil {
        // 输出:[]
        fmt.Println(numbers)
    }
}


切片的切割

在Go语言中,可以使用切片的切片操作来切割切片。切片的切片操作可以用来获取切片的一部分,或者将一个切片分割成多个子切片。切片的切片操作使用两个索引来指定切割的位置:第一个索引指定切割的起始位置,第二个索引指定切割的结束位置,但不包括该位置的元素。第一个索引不指定时,默认为0。第二个索引不指定时,默认为切片的长度。

package main
import "fmt"
func main() {
    numbers := []int{1, 2, 3, 4, 5, 6}
    // 输出:[3 4 5 6]
    fmt.Println(numbers[2:])
    // 输出:[1 2 3 4]
    fmt.Println(numbers[:4])
    // 输出:[3 4]
    fmt.Println(numbers[2:4])
    // 输出:[1 2 3 4 5 6]
    fmt.Println(numbers[:])
}



切片的添加

可以使用append()函数向切片添加一个或多个元素。append()函数会根据切片的容量和长度,自动调整底层数组的大小,并将新元素添加到切片的末尾。注意:append()函数会返回一个新的切片,包含添加元素后的结果;因此,需要将返回的结果重新赋值给原切片,以更新切片的内容。

package main
import "fmt"
func main() {
    numbers := []int{1, 2, 3}
    // 添加元素99和100到切片的末尾
    numbers = append(numbers, 99, 100)
    // 输出:[1 2 3 99 100]
    fmt.Println(numbers)
}


当向一个切片添加另一个切片的所有元素时,使用append()函数需要对第二个切片进行解包(切片后面添加符号...,用于展开切片中的元素),可参考下面的示例代码。

package main
import "fmt"
func main() {
    numbers := []int{1, 2, 3}
    numbers2 := []int{99, 100}
    // 添加另一个切片,需要解包
    numbers = append(numbers, numbers2...)
    // 输出:[1 2 3 99 100]
    fmt.Println(numbers)
}


切片的删除

切片的删除分为几种情况:从头部删除、从中间删除、从尾部删除。

删除开头的元素时,可以直接移动数据指针。假如有一个切片slice,则slice[1:]会删除开头1个元素,slice[N:]会删除开头N个元素。

package main
import "fmt"
func main() {
    numbers := []int{1, 2, 3, 4, 5}
    // 删除开头2个元素
    numbers = numbers[2:]
    // 输出:[3 4 5]
    fmt.Println(numbers)
}


也可以不移动数据指针,但将后面的数据向开头移动,此时可用append()函数原地完成。

package main
import "fmt"
func main() {
    numbers := []int{1, 2, 3, 4, 5}
    // 删除开头2个元素
    numbers = append(numbers[:0], numbers[2:]...)
    // 输出:[3 4 5]
    fmt.Println(numbers)
}



删除中间的元素时,需要对剩余元素进行一次整体移动,此时仍可用append()函数原地完成。

package main
import "fmt"
func main() {
    numbers := []int{1, 2, 3, 4, 5, 6}
    // 删除中间的3、4两个元素
    numbers = append(numbers[:2], numbers[4:]...)
    // 输出:[1 2 5 6]
    fmt.Println(numbers)
}


删除尾部的元素时,直接切割即可。

package main
import "fmt"
func main() {
    numbers := []int{1, 2, 3, 4, 5, 6}
    // 删除尾部的两个元素
    numbers = numbers[:len(numbers) - 2]
    // 输出:[1 2 3 4]
    fmt.Println(numbers)
}


切片的复制

可以使用copy()函数来复制切片,它接受两个参数:第一个参数是目标切片,第二个参数是源切片。copy()函数会将源切片中的元素复制到目标切片中,并返回实际复制的元素个数。注意:如果源切片和目标切片不一样大,则会按照其中较小的那个切片的元素个数进行复制。另外,切片的复制不是在末尾添加元素,而是从开头位置覆盖之前已经存在的元素。

package main
import "fmt"
func main() {
    slice1 := []int{1, 2, 3, 4, 5}
    slice2 := []int{66, 88, 99}
    // slice2的容量小,故只会复制slice1的前3个元素到slice2中
    copy(slice2, slice1)
    // 输出:[1 2 3]
    fmt.Println(slice2)
    slice2 = []int{66, 88, 99}
    copy(slice1, slice2)
    // 输出:[66 88 99 4 5]
    fmt.Println(slice1)
}



切片的遍历

有两种方法来实现切片的遍历:一是使用for循环,二是使用range关键字。

在下面的示例代码中,我们使用了一个for循环,从切片的索引0开始遍历到切片的长度减1。通过索引,我们可以访问切片中的每个元素,并打印出来。

package main
import "fmt"
func main() {
    numbers := []int{1, 2, 3, 4, 5}
    for i := 0; i < len(numbers); i++ {
        fmt.Println(numbers[i])
    }
}




在下面的示例代码中,我们使用了range关键字来遍历切片。每次迭代时,range会返回当前元素的索引和值。我们可以使用它们来访问切片中的元素,这种方法更加简洁和易读。

package main
import "fmt"
func main() {
    numbers := []int{1, 2, 3, 4, 5}
    for index, value := range numbers {
        fmt.Println(index, value)
    }
}


多维切片

多维切片是由一系列切片组成的,每个切片可以具有不同的长度和容量。要创建一个多维切片,首先需要创建一个初始切片。然后使用该切片来创建额外的切片,每个新切片都会将原始切片的元素拆分为更小的切片。

声明一个多维切片的语法格式如下:

 var sliceName [][]...[]Type

其中,sliceName表示切片的名称,Type表示切片的类型,每个[]代表着一个维度,切片有几个维度就需要几个[]。

在下面的示例代码中,我们首先创建了一个初始切片slice,长度为3。然后,我们使用range循环为每个内部切片创建了长度为2的子切片。接下来,我们给多维切片中的元素进行了赋值。最后,我们使用嵌套的range循环打印了多维切片的内容。


package main
 
import "fmt"
 
func main() {
    // 创建一个初始切片,长度为3
    slice := make([][]int, 3)
    for i := range slice {
        // 每个内部切片长度为2
        slice[i] = make([]int, 2)
    }
 
    // 给多维切片中的元素赋值
    slice[0][0] = 50
    slice[0][1] = 60
    slice[1][0] = 70
    slice[1][1] = 80
    slice[2][0] = 90
    slice[2][1] = 100
 
    // 打印多维切片的内容
    for i := range slice {
        for j := range slice[i] {
            fmt.Println(slice[i][j])
        }
    }
}


相关文章
|
2月前
|
Go
Go to Learn Go之命令行参数
Go to Learn Go之命令行参数
27 8
|
2月前
|
Serverless Go
Go to Learn Go之时间日期
Go to Learn Go之时间日期
38 8
|
2月前
|
Go
Go to Learn Go之Gob
Go to Learn Go之Gob
22 8
|
2月前
|
Go
Go to Learn Go之文件操作
Go to Learn Go之文件操作
24 8
|
2月前
|
Go
Go to Learn Go之反射
Go to Learn Go之反射
38 8
|
2月前
|
存储 安全 Go
Go to Learn Go之并发
Go to Learn Go之并发
28 8
|
2月前
|
存储 Go
Go to Learn Go之类型转换
Go to Learn Go之类型转换
35 7
|
12天前
|
存储 安全 Go
Go语言切片:从入门到精通的深度探索###
本文深入浅出地剖析了Go语言中切片(Slice)这一核心概念,从其定义、内部结构、基本操作到高级特性与最佳实践,为读者提供了一个全面而深入的理解。通过对比数组,揭示切片的灵活性与高效性,并探讨其在并发编程中的应用优势。本文旨在帮助开发者更好地掌握切片,提升Go语言编程技能。 ###
|
1天前
|
编译器 Go
go语言编译选项
【10月更文挑战第17天】
9 5
|
4天前
|
安全 Go 开发者
go语言并发模型
【10月更文挑战第16天】
18 8