一篇文章带你了解Go语言基础之切片补充

开发 前端
其实切片的本质,还是数组,只不过是Go帮助我们做了一些封装,可以方便的对切片里面的数据增删改查。

前言

Hey,大家好呀,我是星期八,这次咱们继续学习Go基础之切片补充扒。

make疑云

我们知道,可以通过make创建切片。

var names = make([]string,10,10)

这句话表示动态创建了一个切片,切片中的元素数量为10个,切片的容量也为10个。

你有疑惑吗???

切片的数量和容量是什么???

他俩什么关系???

切片本质

其实切片,终究是一个存储数据的一个东西,目前知道数组是可以存储东西的。

其实切片的本质,还是数组,只不过是Go帮助我们做了一些封装,可以方便的对切片里面的数据增删改查。

例如:

package main

import "fmt"

func main() {
var names = make([]int410)
//int类型默认值是0
fmt.Println(names, len(names), cap(names)) //结果:[0 0 0 0] 4 10
}

理解图。

没错,本质就是指向了一个长一点的数组。

但是这个数组是会自动扩容的,当容量(cap)append满了之后,会自动扩容。

现在,我们就知道make里面参数的意义了。

注意:在Go中,推荐使用make创建切片,并且在创建时,需要考虑容量,尽可能不触发容量自动扩容机制,提高性能。

为什么切片append之后,前面会有空格

在上一章中,大概有这样一段代码。


package main

import "fmt"

func main() {
var names = make([]int,5,10)
names = append(names,11,23,231)
fmt.Println(names)//[0 0 0 0 0 11 23 231]
}

append之后,前面会有很多0,这是怎么回事。

解释:

在通过make创建切片时,第二个参数是切片元素的数量。

上述代码切片第二个参数是5,表示在创建切片时,前5个就已经有值了,只不过是int默认值0。

所以再append时,是再原有的基础上,添加值的,直到cap满了之后,触发扩容机制。

如图所示。

现在,清晰了吧?

那怎么append时,从0开始呢???

这不是很简单,直接让第二个参数为0。

var names = make([]int,0,10)
//结果:[11 23 231]

如图所示。

好了,这个,懂了吧,怎么继续哈。

为什么不推荐使用var []类型方式创建切片

我们上述一直在提一个词,自动扩容。

我们来看这样一段普通的代码。

package main

import "fmt"

func main() {
var names []int
//地址:0x0,长度(len):0,容量(cap):0
fmt.Printf("地址:%p,长度(len):%d,容量(cap):%d\n", names, len(names), cap(names))
names = append(names, 123)

//地址:0xc000010380,长度(len):3,容量(cap):4
fmt.Printf("地址:%p,长度(len):%d,容量(cap):%d\n", names, len(names), cap(names))
}

虽然按照这种方法,使用append动态添加是没问题的。

在不使用make声明数组时,len和cap都是0,并且地址也是一个值。

通过append之后,可以明显看到,地址发生了改变,因为又重新申请了数组,切片重新指向新的数组。

len和cap也发生了变化。

copy复制切片

package main

import "fmt"

func main() {
var names1 = make([]string, 010)
names1 = append(names1, "张三")
names1 = append(names1, "李四")
var names2 = names1 //将names1赋值到names2
fmt.Println(names1, names2) //[张三 李四] [张三 李四]
names1[0] = "张三666"//修改names下标为0的值为 张三666
fmt.Println(names1, names2) //[张三666 李四] [张三666 李四]
//为什么修改names1的值,会影响names2的值????
}

为什么修改names1的值,会影响names2的值???

这个,就又要回到内存分布图了,如图所示。

我们说过很多次,不管是打印,还是赋值等操作,只会操作栈上面存储的值。

当names2=names1时,只会把names1栈上面的地址,给names2。

但是存的时堆上面的地址,终究还是指向了同一个堆。

所以修改names1时,names2也修改了。

那如果不想出现上述问题怎么办???

解决办法:使用copy

package main

import "fmt"

func main() {
var names1 = make([]string, 010)
names1 = append(names1, "张三")
names1 = append(names1, "李四")
//定义一个names2切片用于接收,第二个参数要留空间,names1里面又几个元素,names2第二个参数也要是几
var names2 = make([]string, 210)
copy(names2, names1)//将names1的值,赋值到names2
fmt.Println(names1, names2) //[张三 李四] [张三 李四]
names1[0] = "张三666"//修改names下标为0的值为 张三666
fmt.Println(names1, names2) //[张三666 李四] [张三 李四]
fmt.Printf("names1地址:%p names2地址:%p\n",names1,names2)
//names1地址:0xc00009a0a0 names2地址:0xc00009a140
}

内存图:

自动扩容机制

非常抱歉,我不会 。。。

总结

上述我们学习了Go基础之切片补充。如果在操作过程中有任务问题,记得在下面的讨论区留言,我们看到会第一时间解决问题。

责任编辑:武晓燕 来源: Go语言进阶学习
相关推荐

2020-10-22 08:33:22

Go语言

2020-11-11 10:52:54

Go语言C语言

2020-11-05 09:58:16

Go语言Map

2022-02-16 10:03:06

对象接口代码

2020-12-07 05:59:02

语言Go接口

2021-10-30 10:43:04

语言Go函数

2021-11-03 10:02:07

Go基础函数

2020-12-27 10:15:44

Go语言channel管道

2021-10-09 07:10:31

Go语言基础

2020-10-25 07:33:13

Go语言

2020-12-09 09:59:32

Go语言技术

2020-10-23 08:38:19

Go语言

2020-12-30 09:04:32

Go语言TCPUDP

2020-10-22 11:15:47

Go语言变量

2021-10-16 10:17:51

Go语言数据类型

2021-09-29 10:00:07

Go语言基础

2021-10-13 10:00:52

Go语言基础

2020-12-23 08:39:11

Go语言基础技术

2021-01-13 08:40:04

Go语言文件操作

2021-02-20 10:06:14

语言文件操作
点赞
收藏

51CTO技术栈公众号