Go 语言 切片的使用(增删改查)

简介: 引言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: %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. 删除元素

2387773-20220325212351039-1583744322.png


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]

 


相关文章
|
11天前
|
运维 监控 算法
监控局域网其他电脑:Go 语言迪杰斯特拉算法的高效应用
在信息化时代,监控局域网成为网络管理与安全防护的关键需求。本文探讨了迪杰斯特拉(Dijkstra)算法在监控局域网中的应用,通过计算最短路径优化数据传输和故障检测。文中提供了使用Go语言实现的代码例程,展示了如何高效地进行网络监控,确保局域网的稳定运行和数据安全。迪杰斯特拉算法能减少传输延迟和带宽消耗,及时发现并处理网络故障,适用于复杂网络环境下的管理和维护。
|
12天前
|
编译器 Go
揭秘 Go 语言中空结构体的强大用法
Go 语言中的空结构体 `struct{}` 不包含任何字段,不占用内存空间。它在实际编程中有多种典型用法:1) 结合 map 实现集合(set)类型;2) 与 channel 搭配用于信号通知;3) 申请超大容量的 Slice 和 Array 以节省内存;4) 作为接口实现时明确表示不关注值。此外,需要注意的是,空结构体作为字段时可能会因内存对齐原因占用额外空间。建议将空结构体放在外层结构体的第一个字段以优化内存使用。
|
23天前
|
测试技术 Go API
Go 切片导致 rand.Shuffle 产生重复数据的原因与解决方案
在 Go 语言开发中,使用切片时由于其底层数据共享特性,可能会引发意想不到的 Bug。本文分析了 `rand.Shuffle` 后切片数据重复的问题,指出原因在于切片是引用类型,直接赋值会导致底层数组共享,进而影响原始数据。解决方案是使用 `append` 进行数据拷贝,确保独立副本,避免 `rand.Shuffle` 影响原始数据。总结强调了切片作为引用类型的特性及正确处理方法,确保代码稳定性和正确性。
115 82
|
17天前
|
存储 Go
Go 语言入门指南:切片
Golang中的切片(Slice)是基于数组的动态序列,支持变长操作。它由指针、长度和容量三部分组成,底层引用一个连续的数组片段。切片提供灵活的增减元素功能,语法形式为`[]T`,其中T为元素类型。相比固定长度的数组,切片更常用,允许动态调整大小,并且多个切片可以共享同一底层数组。通过内置的`make`函数可创建指定长度和容量的切片。需要注意的是,切片不能直接比较,只能与`nil`比较,且空切片的长度为0。
Go 语言入门指南:切片
|
20天前
|
算法 安全 Go
公司局域网管理系统里的 Go 语言 Bloom Filter 算法,太值得深挖了
本文探讨了如何利用 Go 语言中的 Bloom Filter 算法提升公司局域网管理系统的性能。Bloom Filter 是一种高效的空间节省型数据结构,适用于快速判断元素是否存在于集合中。文中通过具体代码示例展示了如何在 Go 中实现 Bloom Filter,并应用于局域网的 IP 访问控制,显著提高系统响应速度和安全性。随着网络规模扩大和技术进步,持续优化算法和结合其他安全技术将是企业维持网络竞争力的关键。
44 2
公司局域网管理系统里的 Go 语言 Bloom Filter 算法,太值得深挖了
|
16天前
|
开发框架 前端开发 Go
eino — 基于go语言的大模型应用开发框架(二)
本文介绍了如何使用Eino框架实现一个基本的LLM(大语言模型)应用。Eino中的`ChatModel`接口提供了与不同大模型服务(如OpenAI、Ollama等)交互的统一方式,支持生成完整响应、流式响应和绑定工具等功能。`Generate`方法用于生成完整的模型响应,`Stream`方法以流式方式返回结果,`BindTools`方法为模型绑定工具。此外,还介绍了通过`Option`模式配置模型参数及模板功能,支持基于前端和用户自定义的角色及Prompt。目前主要聚焦于`ChatModel`的`Generate`方法,后续将继续深入学习。
132 7
|
12天前
|
存储 缓存 监控
企业监控软件中 Go 语言哈希表算法的应用研究与分析
在数字化时代,企业监控软件对企业的稳定运营至关重要。哈希表(散列表)作为高效的数据结构,广泛应用于企业监控中,如设备状态管理、数据分类和缓存机制。Go 语言中的 map 实现了哈希表,能快速处理海量监控数据,确保实时准确反映设备状态,提升系统性能,助力企业实现智能化管理。
27 3
|
12天前
|
存储 缓存 安全
Go 语言中的 Sync.Map 详解:并发安全的 Map 实现
`sync.Map` 是 Go 语言中用于并发安全操作的 Map 实现,适用于读多写少的场景。它通过两个底层 Map(`read` 和 `dirty`)实现读写分离,提供高效的读性能。主要方法包括 `Store`、`Load`、`Delete` 等。在大量写入时性能可能下降,需谨慎选择使用场景。
|
27天前
|
监控 Linux PHP
【02】客户端服务端C语言-go语言-web端PHP语言整合内容发布-优雅草网络设备监控系统-2月12日优雅草简化Centos stream8安装zabbix7教程-本搭建教程非docker搭建教程-优雅草solution
【02】客户端服务端C语言-go语言-web端PHP语言整合内容发布-优雅草网络设备监控系统-2月12日优雅草简化Centos stream8安装zabbix7教程-本搭建教程非docker搭建教程-优雅草solution
79 20
|
17天前
|
存储 开发框架 Devops
eino — 基于go语言的大模型应用开发框架(一)
Eino 是一个受开源社区优秀LLM应用开发框架(如LangChain和LlamaIndex)启发的Go语言框架,强调简洁性、可扩展性和可靠性。它提供了易于复用的组件、强大的编排框架、简洁明了的API、最佳实践集合及实用的DevOps工具,支持快速构建和部署LLM应用。Eino不仅兼容多种模型库(如OpenAI、Ollama、Ark),还提供详细的官方文档和活跃的社区支持,便于开发者上手使用。
107 8

热门文章

最新文章

相关实验场景

更多