【Go】slice

2024-05-24 21:38
文章标签 go slice

本文主要是介绍【Go】slice,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

go里面没有特别好用的list类型,查了一下有一个container/list包可以用,不过感觉用的人少,一般大家都直接用slice。决定看下slice。

slice和map一样,是一个引用类型。概念上可以看做一个结构体,里面有三个成员,一个是数组的指针,一个是len,另一个是容量cap。

slice是一个动态数组,但是是轻量级的,内部包含了一个底层数组的指针,一个len用于表示slice的大小,容量则是底层数组的大小,当然需要从slice的起始位置算起。

(1)访问一个slice时,不可以超过len的大小,否则会报一个panic 越界错误。如果需要扩容,只能通过append函数。

(2)切片操作不能超过cap的大小,否则也会报panic 错误。

由于slice是引用类型,当多个slice由同一个底层数组切片创建,那么他们很可能有公共部分,一个的修改会影响到其他。这是需要注意的。

slice的append函数用于追加元素,实现逻辑大致是先看底层数组的cap够不够,够的话直接追加到原数组,否则新建一个大的底层数组,然后返回,所以每一次的append,我们都需要赋值。具体append有没有创建新的底层数组不清楚,这个没法保证,取决于go底层内存的管理分配。

make函数可以创建一个slice,可以指定底层数组的cap和len。所以目前为止,我们知道了slice的两种创建方式,一种是从其他的数组或者slice使用切片,或者使用make创建新的。

copy函数,把第二个参数的slice复制到第一个参数的slice,赋值的数目是二者len的最小值。这个函数很有用,比如我们自己需要新定义一个在第一个位置追加元素的函数,那么就需要copy函数了。

比如:

func addFirst(l []int, x int) []int {newL := make([]int, len(l)+1)newL[0] = xcopy(newL[1:], l)return newL
}
这个实现,每一次都会新建一个底层数组,因为用的是make函数。如果更好一点可以判断需不需要创建新的数组,如果cap够,那么只需要后移后面的元素即可。

这部分逻辑可以让append函数来做:

func addFirst(l []int, x int) []int {l = append(l, 999)copy(l[1:], l[:len(l)-1])l[0] = xreturn l
}
所以,go里面slice还是很有用的,需要明白其底层是怎么实现的,然后就是配合copy,make和appned函数来使用。


这篇关于【Go】slice的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/999605

相关文章

Go信号处理如何优雅地关闭你的应用

《Go信号处理如何优雅地关闭你的应用》Go中的优雅关闭机制使得在应用程序接收到终止信号时,能够进行平滑的资源清理,通过使用context来管理goroutine的生命周期,结合signal... 目录1. 什么是信号处理?2. 如何优雅地关闭 Go 应用?3. 代码实现3.1 基本的信号捕获和优雅关闭3.2

Go Playground 在线编程环境

For all examples in this and the next chapter, we will use Go Playground. Go Playground represents a web service that can run programs written in Go. It can be opened in a web browser using the follow

go基础知识归纳总结

无缓冲的 channel 和有缓冲的 channel 的区别? 在 Go 语言中,channel 是用来在 goroutines 之间传递数据的主要机制。它们有两种类型:无缓冲的 channel 和有缓冲的 channel。 无缓冲的 channel 行为:无缓冲的 channel 是一种同步的通信方式,发送和接收必须同时发生。如果一个 goroutine 试图通过无缓冲 channel

如何确定 Go 语言中 HTTP 连接池的最佳参数?

确定 Go 语言中 HTTP 连接池的最佳参数可以通过以下几种方式: 一、分析应用场景和需求 并发请求量: 确定应用程序在特定时间段内可能同时发起的 HTTP 请求数量。如果并发请求量很高,需要设置较大的连接池参数以满足需求。例如,对于一个高并发的 Web 服务,可能同时有数百个请求在处理,此时需要较大的连接池大小。可以通过压力测试工具模拟高并发场景,观察系统在不同并发请求下的性能表现,从而

【Go】go连接clickhouse使用TCP协议

离开你是傻是对是错 是看破是软弱 这结果是爱是恨或者是什么 如果是种解脱 怎么会还有眷恋在我心窝 那么爱你为什么                      🎵 黄品源/莫文蔚《那么爱你为什么》 package mainimport ("context""fmt""log""time""github.com/ClickHouse/clickhouse-go/v2")func main(

Go Select的实现

select语法总结 select对应的每个case如果有已经准备好的case 则进行chan读写操作;若没有则执行defualt语句;若都没有则阻塞当前goroutine,直到某个chan准备好可读或可写,完成对应的case后退出。 Select的内存布局 了解chanel的实现后对select的语法有个疑问,select如何实现多路复用的,为什么没有在第一个channel操作时阻塞 从而导

Go Channel的实现

channel作为goroutine间通信和同步的重要途径,是Go runtime层实现CSP并发模型重要的成员。在不理解底层实现时,经常在使用中对channe相关语法的表现感到疑惑,尤其是select case的行为。因此在了解channel的应用前先看一眼channel的实现。 Channel内存布局 channel是go的内置类型,它可以被存储到变量中,可以作为函数的参数或返回值,它在r

Go 数组赋值问题

package mainimport "fmt"type Student struct {Name stringAge int}func main() {data := make(map[string]*Student)list := []Student{{Name:"a",Age:1},{Name:"b",Age:2},{Name:"c",Age:3},}// 错误 都指向了最后一个v// a

Go组合

摘要 golang并非完全面向对象的程序语言,为了实现面向对象的继承这一神奇的功能,golang允许struct间使用匿名引入的方式实现对象属性方法的组合 组合使用注意项 使用匿名引入的方式来组合其他struct 默认优先调用外层方法 可以指定匿名struct以调用内层方法 代码 package mainimport ("fmt")type People struct{}type Pe

Go语言构建单链表

package mainimport "fmt"type ListNode struct {Val intNext *ListNode}func main() {list := []int{2,4,3}head := &ListNode{Val:list[0]}tail := head //需要头尾两个指针for i:=1;i<len(list);i++ {//方法一 数组直接构建链表tai