【Go语言】【7】GO语言的切片

简介:
原创作品,允许转载,转载时请务必以超链接形式标明文章  原始出处 、作者信息和本声明。否则将追究法律责任。 http://qingkechina.blog.51cto.com/5552198/1616987

如果说GO语言的数组为静态长度的数组,那么切片(slice)则为动态长度的数组

一、基于数组创建切片

1、存在一个整型数组intArr := [10]int{1, 2, 3, 4, 5, 6, 7, 8, 9, 10},那么下面的slice就是数组切片

var slice []int = intArr[3:7]

wKiom1TzEKvQ5lSYAABDmRP4YK8764.jpg

从图中可以看出切片从数组的第4个元素开始读取数据,直至第8个元素(但不包含第8个)。切记程序员的计数都是从0开始的哟i_f30.gif


2、若只读intArr数组的前4个元素,该如何办呢?聪明的你一定能想到

var slice [] int = intArr[0:4]

或者

slice := intArr[0:4]


3、是否还有更简洁的表示吗?那肯定了,GO这么懂程序员

slice := intArr[:4],它与slice := intArr[0:4]等价


4、哪些读取intArr数组第4个元素(包含第4个元素)之后的所有数据呢?

slice := intArr[3:10]

当然也可以如下表示

slice := intArr[3:len(intArr)]

或者

slice := intArr[3:]


二、基于数组创建的切片与该数组的关系

上例中基于数组intArr创建了切片slice,那么slice与intArr有什么关系呢?可以简单地理解为切片是在数组的基础上增加了一些管理功能,很类型C++中的数组与std::vector的关系。

intArr := [10]int{1,2,3,4,5,6,7,8,9,10} // 定义数组intArr

var slice []int = intArr[3:7]        // 基于intArr创建切片 


fmt.Printf("intArr的地址是: %p\n", &intArr) 

fmt.Printf("slice的地址是 : %p\n", &slice)


fmt.Printf("intArr的第4个元素地址是: %p\n", &intArr[3])

fmt.Printf("slice的第1个元素地址是 : %p\n", &slice[0])wKiom1T0Vy7QGnb7AANZonkIBnU543.jpg

从程序运行结果上可以看切片slice和数组intArr两者没有任何关系,是不同的内在地址;但有意思的是切片slice的第一个元素地址恰恰是数组intArr的第4个元素地址,这不是巧合,而是切片的实质

wKiom1T0agKDLE86AADJOhQDpwA427.jpg

所以有人也说切片是指向数组的指针,对切片元素的更改会影响数组元素的值,这个通过改写上面示例可以得到验证:

intArr := [10]int{1, 2, 3, 4, 5, 6, 7, 8, 9, 10}

slice := intArr[3:]

slice[0] = 100 // 更改切片的第1个元素值为100

fmt.Println("slice=", slice)

fmt.Println("intArr=", intArr)

wKiom1T0awuBcatWAABUvGIMtF0935.jpg


三、基于切片创建切片

能否基于切片创建切片呢?答案是肯定的,其使用方法与基于数组创建切片相同

intArr := [10]int{1, 2, 3, 4, 5, 6, 7, 8, 9, 10}

slice := intArr[3:]

reslice := slice[1:]

fmt.Println(reslice)

wKioL1T1oObxjV1yAAAmMpP3f9I678.jpg

由于slice是基于数组intArr创建的切片,它从intArr的第4个值开始,所以slice的值为[4,5,6,7,8,9,10]; 而reslice是基于切片slice创建的切片,它从slice的第2个值开始,所以reslice的值为[5,6,7,8,9,10]


四、直接创建切片

并不是非得基于数组或者切片才能创建切片,GO语言很贴心地为程序员提供了make()函数,其形式为slice := make([] type, number, capacity),例如:

slice := make([]int, 3, 10)

1、number和capacity分别是什么意思?

number是切片元素个数,创建切片slice之后,里面有几个元素;capacity表示切片的容量,创建切片slice之后,先预留10个元素的存储空间。运行上面例子便能理解:

wKioL1T1pQPSLsJfAABYae7tmek339.jpg

从结果上来看,切片slice元素个数为3,且元素缺省填充为0;由于切片slice的容量为10,所以还可以向切片追加7个元素。


2、len()和cap()

切片有len()和cap()函数,分别用来获取切片的元素个数和容量值,例如:

slice := make([]int, 3, 10)

num := len(slice)

capacity := cap(slice)

fmt.Println("切片slice元素个数为:", num, ",容量为:", capacity)

wKiom1T1pZyhU9tVAABCLSlHOTc784.jpg


3、能省略容量吗?

答案是OK的,所以可以这样声明切片:slice := make([]int,3)

那么此时它的容量与元素个数相同,即为3


4、能把元素个数和容量都省略变为slice := make([]int)可行吗?

答案是NO,此时GO语言就晕了,它在想难道程序员让我把所有内存都占用?就抛出missing len argument to make([]int)错误信息


5、只有类型和容量,省略元素个数呢,类似slice := make([]int,,10)

答案是NOj_0081.gif,但可以声明后直接初始化,即slice := []int{1,2,3,4,5,6}y_0003.gif


6、如何追加元素?

slice := make([]int, 3, 10)

fmt.Println("切片slice元素个数为:", len(slice), ",容量为:", cap(slice), ",slice=", slice)

可以这样追加:

slice = append(slice, 4)

slice = append(slice, 5)

也可以这样追加:

slice = append(slice, 6, 7, 8)

甚至可以追加一个切片,但在后面必须添加...

arr := [2]int{9, 10}

slice = append(slice, arr...)

程序运行追加后的结果如下:

wKiom1T1qsjCox0EAAMP71W_5V0528.jpg


五、切片追加元素后切片是否重新生成?

1、追加说明

slice := make([]int, 3, 10) // 声明一个切片

fmt.Printf("原切片的地址%p\n", &slice)

fmt.Printf("原切片第一个元素的地址%p\n", &slice[0])


slice = append(slice, 4)  // 追加一个元素

fmt.Printf("新切片的地址%p\n", &slice)

fmt.Printf("新切片第一个元素的地址%p\n", &slice[0])

wKiom1T1r7PDJAjOAACdEk52urY469.jpg

从运行结果上可以看到,追加一个元素后,切片元素的地址并没有发生变化;但在应用时必须返回值赋给slice,即下面的程序是错误的:

slice := make([]int, 3, 10) 

append(slice, 4) // 必须返回slice

fmt.Printf("%p\n", &slice)


2、追加超过容量大小

slice := make([]int, 3, 10)                   // 切片容量为10

fmt.Printf("%p\n", &slice)                    // 打印切片地址

fmt.Println("len:", len(slice), "cap:", cap(slice))   // 打印切片元素个数和容量


slice = append(slice, 4, 5, 6, 7, 8, 9, 10, 11)      // 切片追加元素

fmt.Printf("%p\n", &slice)                    // 打印切片地址

fmt.Println("len:", len(slice), "cap:", cap(slice))   // 打印切片元素个数和容量

这个例子中一开始声明了容量为10的切片,当向切片中追加元素时,使其元素个数超过容量,此时切片会自动扩展容量一倍变为20,这点与Java的集合很相似
wKiom1T1rgrCXhDwAALhkil8MXE502.jpg

本文出自 “青客” 博客,请务必保留此出处http://qingkechina.blog.51cto.com/5552198/1616987

目录
相关文章
|
1天前
|
Java Go
一文带你速通go语言指针
Go语言指针入门指南:简述指针用于提升效率,通过地址操作变量。文章作者sharkChili是Java/CSDN专家,维护Java Guide项目。文中介绍指针声明、取值,展示如何通过指针修改变量值及在函数中的应用。通过实例解析如何使用指针优化函数,以实现对原变量的直接修改。作者还邀请读者加入交流群深入探讨,并鼓励关注其公众号“写代码的SharkChili”。
8 0
|
1天前
|
存储 缓存 Java
来聊聊go语言的hashMap
本文介绍了Go语言中的`map`与Java的不同设计思想。作者`sharkChili`是一名Java和Go开发者,同时也是CSDN博客专家及JavaGuide项目的维护者。文章探讨了Go语言`map`的数据结构,包括`count`、`buckets指针`和`bmap`,解释了键值对的存储方式,如何利用内存对齐优化空间使用,并展示了`map`的初始化、插入键值对以及查找数据的源码过程。此外,作者还分享了如何通过汇编查看`map`操作,并鼓励读者深入研究Go的哈希冲突解决和源码。最后,作者提供了一个交流群,供读者讨论相关话题。
9 0
|
2天前
|
Java Go
Go语言学习11-数据初始化
【5月更文挑战第3天】本篇带大家通过内建函数 new 和 make 了解Go语言的数据初始化过程
16 1
Go语言学习11-数据初始化
|
2天前
|
自然语言处理 安全 Java
速通Go语言编译过程
Go语言编译过程详解:从词法分析(生成token)到句法分析(构建语法树),再到语义分析(类型检查、推断、匹配及函数内联)、生成中间码(SSA)和汇编码。最后,通过链接生成可执行文件。作者sharkchili,CSDN Java博客专家,分享技术细节,邀请读者加入交流群。
20 2
|
3天前
|
Java Linux Go
一文带你速通Go语言基础语法
本文是关于Go语言的入门介绍,作者因其简洁高效的特性对Go语言情有独钟。文章首先概述了Go语言的优势,包括快速上手、并发编程简单、设计简洁且功能强大,以及丰富的标准库。接着,文章通过示例展示了如何编写和运行Go代码,包括声明包、导入包和输出语句。此外,还介绍了Go的语法基础,如变量类型(数字、字符串、布尔和复数)、变量赋值、类型转换和默认值。文章还涉及条件分支(if和switch)和循环结构(for)。最后,简要提到了Go函数的定义和多返回值特性,以及一些常见的Go命令。作者计划在后续文章中进一步探讨Go语言的其他方面。
9 0
|
4天前
|
JavaScript 前端开发 Go
Go语言的入门学习
【4月更文挑战第7天】Go语言,通常称为Golang,是由Google设计并开发的一种编程语言,它于2009年公开发布。Go的设计团队主要包括Robert Griesemer、Rob Pike和Ken Thompson,这三位都是计算机科学和软件工程领域的杰出人物。
12 1
|
4天前
|
Go
|
5天前
|
分布式计算 Java Go
Golang深入浅出之-Go语言中的分布式计算框架Apache Beam
【5月更文挑战第6天】Apache Beam是一个统一的编程模型,适用于批处理和流处理,主要支持Java和Python,但也提供实验性的Go SDK。Go SDK的基本概念包括`PTransform`、`PCollection`和`Pipeline`。在使用中,需注意类型转换、窗口和触发器配置、资源管理和错误处理。尽管Go SDK文档有限,生态系统尚不成熟,且性能可能不高,但它仍为分布式计算提供了可移植的解决方案。通过理解和掌握Beam模型,开发者能编写高效的数据处理程序。
134 1
|
5天前
|
算法 关系型数据库 MySQL
Go语言中的分布式ID生成器设计与实现
【5月更文挑战第6天】本文探讨了Go语言在分布式系统中生成全局唯一ID的策略,包括Twitter的Snowflake算法、UUID和MySQL自增ID。Snowflake算法通过时间戳、节点ID和序列号生成ID,Go实现中需处理时间回拨问题。UUID保证全局唯一,但长度较长。MySQL自增ID依赖数据库,可能造成性能瓶颈。选择策略时需考虑业务需求和并发、时间同步等挑战,以确保系统稳定可靠。
112 0
|
5天前
|
缓存 NoSQL Go
Go语言中的分布式锁实现与选型
【5月更文挑战第6天】本文探讨了Go语言中分布式锁的实现,包括Redis、ZooKeeper和Etcd三种方式,强调了选型时的性能、可靠性和复杂度考量。通过代码示例展示了Redis分布式锁的使用,并提出了避免死锁、公平性等问题的策略。结论指出,开发者应根据业务需求选择合适实现并理解底层原理,以确保系统稳定和高效。
132 0