go语言的切片研究

简介: go语言的切片研究

切片slice

Go 语言切片是对数组的抽象。

Go 数组的长度不可改变,在特定场景中这样的集合就不太适用,Go 中提供了一种灵活,功能强悍的内置类型切片("动态数组"),与数组相比切片的长度是不固定的,可以追加元素,在追加时可能使切片的容量增大。

声明切片

package main
import "fmt"
func main() {
   //1:direct declare variable
   var s1  \[\]int
   //s1 = \[\]int{}
   fmt.Printf("s1: %v,len: %v,cap: %v \\n",s1,len(s1),cap(s1))
   //2:assignment variable
   s2  := \[\]int{1,2,3}
   fmt.Printf("s2: %v,len: %v,cap: %v \\n",s2,len(s2),cap(s2))
   //3:make slice
   s3 :=make(\[\]int,0)
   fmt.Printf("s3 %v,len: %v,cap: %v \\n",s3,len(s3),cap(s3))
   //3:make slice and assignment slice length
   s4 :=make(\[\]int,0,10)
   fmt.Printf("s4: %v,len: %v,cap: %v \\n",s4,len(s4),cap(s4))
   //4:slice by array
   arr := \[5\]int{1,2,3,4,5}
   s5 := arr\[:\]
   fmt.Printf("s5: %v,len: %v,cap: %v \\n",s5,len(s5),cap(s5))
   s6 := arr\[1:len(arr)-1\]
   fmt.Printf("s6: %v,len: %v,cap: %v \\n",s6,len(s6),cap(s6))
   //5:slice by slice
   s7 := \[\]int{1,2,3,4,5,6,7,8,9,10}
   fmt.Printf("s7: %v,len: %v,cap: %v \\n",s7,len(s7),cap(s7))
   s8 := s7\[1:len(arr)-1\]
   fmt.Printf("s8: %v,len: %v,cap: %v \\n",s8,len(s8),cap(s8))
   //\[x:y:z\]  x:start index y:end index z:slice cap=z-x
   s9 := s7\[6:10:10\]
   fmt.Printf("s9: %v,len: %v,cap: %v \\n",s9,len(s9),cap(s9))
}

切片cap,len

cap


切片容量,在追加切片时,可能会使得切片容量变大

len

切片长度,表示切片数组的长度

package main
import "fmt"
func main() {
   var s1  \[\]int
   fmt.Printf("s1: %v,len: %v,cap: %v \\n",s1,len(s1),cap(s1))
   s1 = \[\]int{1,2,3}
   fmt.Printf("s1: %v,len: %v,cap: %v \\n",s1,len(s1),cap(s1))
   var arr =\[\]int{1,2,3,4,5,6,7,8,9,10}
   s2 := arr\[0:5\]
   fmt.Printf("s2: %v,len: %v,cap: %v \\n",s2,len(s2),cap(s2))
}

输出:

image.png

可以看到,在数组中获取切片后,cap=10,len=5

切片的cap永远是大于等于len的

空nil切片

切片未初始化之前,切片等于nil,len和cap都为0

操作切片

append 切片追加数据

package main
import (
   "fmt"
)
func main() {
   var s1 = make(\[\]int,0,10)
   fmt.Printf("s1: %v,len: %v,cap: %v \\n",s1,len(s1),cap(s1))
   s1 = append(s1,1,2,3,4 )
   fmt.Printf("s1: %v,len: %v,cap: %v \\n",s1,len(s1),cap(s1))
}

当追加的数据超出cap容量时,将会触发切片重新分配底层数组,即时原数组并未被填满

重新分配底层数组时,一般以原有cap进行二倍扩容分配

package main
import (
   "fmt"
)
func main() {
   var s1 = make(\[\]int,0,4)
   fmt.Printf("s1: %v,len: %v,cap: %v \\n",s1,len(s1),cap(s1))
   s1 = append(s1,1,2,3,4)
   fmt.Printf("1 s1 pointer: %p \\n",s1)
   fmt.Printf("s1: %v,len: %v,cap: %v \\n",s1,len(s1),cap(s1))
   fmt.Printf("2 s1 pointer: %p \\n",s1)
   s1 = append(s1,5,6)
   fmt.Printf("s1: %v,len: %v,cap: %v \\n",s1,len(s1),cap(s1))
   fmt.Printf("3 s1 pointer: %p \\n",s1)
}

注意:由于每次超出时都会重新分配数组,在开发时应该避免重新分配,一次性定义好切片的容量

copy

copy函数可以将一个切片的元素复制到另一个切片,复制的长度由最小的切片长度为准:

package main
import (
   "fmt"
)
func main() {
   var s1 = make(\[\]int, 0, 4)
   var s2 = make(\[\]int, 0, 10)
   s1 = append(s1, 9)
   s2 = append(s2, 1, 2, 3, 4, 5, 6)
   copy(s2,s1)
   fmt.Printf("s1: %v,len: %v,cap: %v \\n", s1, len(s1), cap(s1))
   fmt.Printf("s2: %v,len: %v,cap: %v \\n", s2, len(s2), cap(s2))
}

输出:

s1: \[9\],len: 1,cap: 4 
s2: \[9 2 3 4 5 6\],len: 6,cap: 10

当s1复制到s2时,s1的元素将完全复制到s2中(如果s2容量比s1长度小,则会忽略s1超出的部分)

切片copy自身

当切片 通过[:x]方式创建新切片时,将直接使用原有切片的地址,同时如果新切片发生了更改,原有切片也将发生更改:

package main
import (
   "fmt"
)
func main() {
   s1 :=\[\]int{1,2,3,4,5,6,7,8,9}
   s2 :=s1\[8:\]
   s3 :=s1\[:8\]
   fmt.Printf("s1 pointer:%p \\n", s1)
   fmt.Printf("s2 pointer:%p \\n", s2)
   fmt.Printf("s3 pointer:%p \\n", s3)
   //s2 pointer == s3 pointer
   copy(s3,s2)
   fmt.Printf("s3: %v,len: %v,cap: %v \\n", s3, len(s3), cap(s3))
   fmt.Printf("s1: %v,len: %v,cap: %v \\n", s1, len(s1), cap(s1))
}

输出:

s1 pointer:0xc00001e0a0 
s2 pointer:0xc00001e0e0 
s3 pointer:0xc00001e0a0 
s3: \[9 2 3 4 5 6 7 8\],len: 8,cap: 9 
s1: \[9 2 3 4 5 6 7 8 9\],len: 9,cap: 9

切片遍历

可直接通过range遍历:

package main
import (
   "fmt"
)
func main() {
   s1 := \[\]int{1, 2, 3, 4, 5, 6, 7, 8, 9}
   fmt.Printf("s1: %v,len: %v,cap: %v \\n", s1, len(s1), cap(s1))
   for key, value := range s1 {
      fmt.Printf("key:%v,value:%v \\n",key,value)
   }
}
目录
相关文章
|
5天前
|
存储 JSON 监控
Viper,一个Go语言配置管理神器!
Viper 是一个功能强大的 Go 语言配置管理库,支持从多种来源读取配置,包括文件、环境变量、远程配置中心等。本文详细介绍了 Viper 的核心特性和使用方法,包括从本地 YAML 文件和 Consul 远程配置中心读取配置的示例。Viper 的多来源配置、动态配置和轻松集成特性使其成为管理复杂应用配置的理想选择。
23 2
|
9天前
|
JavaScript Java Go
探索Go语言在微服务架构中的优势
在微服务架构的浪潮中,Go语言以其简洁、高效和并发处理能力脱颖而出。本文将深入探讨Go语言在构建微服务时的性能优势,包括其在内存管理、网络编程、并发模型以及工具链支持方面的特点。通过对比其他流行语言,我们将揭示Go语言如何成为微服务架构中的一股清流。
|
3天前
|
Go 索引
go语言中的循环语句
【11月更文挑战第4天】
11 2
|
3天前
|
Go C++
go语言中的条件语句
【11月更文挑战第4天】
14 2
|
8天前
|
Ubuntu 编译器 Linux
go语言中SQLite3驱动安装
【11月更文挑战第2天】
30 7
|
8天前
|
关系型数据库 Go 网络安全
go语言中PostgreSQL驱动安装
【11月更文挑战第2天】
38 5
|
8天前
|
安全 Go
用 Zap 轻松搞定 Go 语言中的结构化日志
在现代应用程序开发中,日志记录至关重要。Go 语言中有许多日志库,而 Zap 因其高性能和灵活性脱颖而出。本文详细介绍如何在 Go 项目中使用 Zap 进行结构化日志记录,并展示如何定制日志输出,满足生产环境需求。通过基础示例、SugaredLogger 的便捷使用以及自定义日志配置,帮助你在实际开发中高效管理日志。
25 1
|
7天前
|
程序员 Go
go语言中的控制结构
【11月更文挑战第3天】
84 58
|
6天前
|
监控 Go API
Go语言在微服务架构中的应用实践
在微服务架构的浪潮中,Go语言以其简洁、高效和并发处理能力脱颖而出,成为构建微服务的理想选择。本文将探讨Go语言在微服务架构中的应用实践,包括Go语言的特性如何适应微服务架构的需求,以及在实际开发中如何利用Go语言的特性来提高服务的性能和可维护性。我们将通过一个具体的案例分析,展示Go语言在微服务开发中的优势,并讨论在实际应用中可能遇到的挑战和解决方案。
|
7天前
|
存储 编译器 Go
go语言中的变量、常量、数据类型
【11月更文挑战第3天】
24 9