深入理解Go语言中的数组和切片

作者:daisy 时间:2024-02-04 05:09:54 

一、类型

数组是值类型,将一个数组赋值给另一个数组时,传递的是一份拷贝。

切片是引用类型,切片包装的数组称为该切片的底层数组。

我们来看一段代码


//a是一个数组,注意数组是一个固定长度的,初始化时候必须要指定长度,不指定长度的话就是切片了
a := [3]int{1, 2, 3}
//b是数组,是a的一份拷贝
b := a
//c是切片,是引用类型,底层数组是a
c := a[:]
for i := 0; i < len(a); i++ {
a[i] = a[i] + 1
}
//改变a的值后,b是a的拷贝,b不变,c是引用,c的值改变
fmt.Println(a) //[2,3,4]
fmt.Println(b) //[1 2 3]
fmt.Println(c) //[2,3,4]

二、make

make 只能用于slice, map channel, 所以下面一段代码生成了一个slice,是引用类型


s1 := make([]int, 0, 3)

for i := 0; i < cap(s1); i++ {
s1 = append(s1, i)
}
s2 := s1
for i := 0; i < len(a); i++ {
s1[i] = s1[i] + 1
}

fmt.Println(s1) //[1 2 3]
fmt.Println(s2) //[1 2 3]

三、当对slice append 超出底层数组的界限时


//n1是n2的底层数组
n1 := [3]int{1, 2, 3}
n2 := n1[0:3]
fmt.Println("address of items in n1: ")
for i := 0; i < len(n1); i++ {
fmt.Printf("%p\n", &n1[i])
}
//address of items in n1:
//0xc20801e160
//0xc20801e168
//0xc20801e170
fmt.Println("address of items in n2: ")
for i := 0; i < len(n2); i++ {
fmt.Printf("%p\n", &n2[i])
}
//address of items in n2:
//0xc20801e160
//0xc20801e168
//0xc20801e170

//对n2执行append操作后,n2超出了底层数组n1的j
n2 = append(n2, 1)
fmt.Println("address of items in n1: ")
for i := 0; i < len(n1); i++ {
fmt.Printf("%p\n", &n1[i])
}
//address of items in n1:
//0xc20801e160
//0xc20801e168
//0xc20801e170

fmt.Println("address of items in n2: ")
for i := 0; i < len(n2); i++ {
fmt.Printf("%p\n", &n2[i])
}
//address of items in n2:
//0xc20803a2d0
//0xc20803a2d8
//0xc20803a2e0
//0xc20803a2e8

四、引用“失效”

实现了删除slice最后一个item的函数


func rmLast(a []int) {
fmt.Printf("[rmlast] the address of a is %p", a)
a = a[:len(a)-1]
fmt.Printf("[rmlast] after remove, the address of a is %p", a)
}

调用此函数后,发现原来的slice并没有改变


func main() {
xyz := []int{1, 2, 3, 4, 5, 6, 7, 8, 9}
fmt.Printf("[main] the address of xyz is %p\n", xyz)
rmLast(xyz)
fmt.Printf("[main] after remove, the address of xyz is %p\n", xyz)
fmt.Printf("%v", xyz) //[1 2 3 4 5 6 7 8 9]
}

打印出来的结果如下:


[main] the address of xyz is 0xc2080365f0
[rmlast] the address of a is 0xc2080365f0
[rmlast] after remove, the address of a is 0xc2080365f0
[main] after remove, the address of xyz is 0xc2080365f0
[1 2 3 4 5 6 7 8 9]

这里直接打印了slice的指针值,因为slice是引用类型,所以指针值都是相同的,我们换成打印slice的地址看下


func rmLast(a []int) {
fmt.Printf("[rmlast] the address of a is %p", &a)
a = a[:len(a)-1]
fmt.Printf("[rmlast] after remove, the address of a is %p", &a)
}
func main() {
xyz := []int{1, 2, 3, 4, 5, 6, 7, 8, 9}
fmt.Printf("[main] the address of xyz is %p\n", &xyz)
rmLast(xyz)
fmt.Printf("[main] after remove, the address of xyz is %p\n", &xyz)
fmt.Printf("%v", xyz) //[1 2 3 4 5 6 7 8 9]
}

结果:


[main] the address of xyz is 0xc20801e1e0
[rmlast] the address of a is 0xc20801e200
[rmlast] after remove, the address of a is 0xc20801e200
[main] after remove, the address of xyz is 0xc20801e1e0
[1 2 3 4 5 6 7 8 9]

这次可以看到slice作为函数参数传入函数时,实际上也是拷贝了一份slice,因为slice本身是个指针,所以从现象来看,slice是引用类型

总结

标签:go语言,数组,切片
0
投稿

猜你喜欢

  • python 实现图像快速替换某种颜色

    2021-07-20 03:16:13
  • 交互设计模式(二)-Pagination(分页,标记页数)

    2009-08-03 13:37:00
  • 一文带你吃透Golang中的类型转换

    2024-02-20 18:12:28
  • python实现简单倒计时功能

    2022-09-29 03:08:32
  • 关于go-zero服务自动收集问题分析

    2024-04-26 17:29:51
  • 完美解决MySQL通过localhost无法连接数据库的问题

    2024-01-26 17:41:10
  • Python列表生成式与生成器操作示例

    2023-08-05 14:16:45
  • mysqld_safe启动脚本源码阅读、分析

    2024-01-22 14:33:21
  • 你喜欢篮球吗?Python实现篮球游戏

    2023-10-26 00:28:36
  • MYSQL必知必会读书笔记第六章之过滤数据

    2024-01-28 19:15:03
  • c#使用FreeSql生产环境时自动升级备份数据库

    2024-01-22 15:56:01
  • 挑战! 纯Javascript 重现经典网游! <魔力宝贝>

    2008-10-04 10:37:00
  • Go开发Gin项目添加jwt功能实例详解

    2024-04-26 17:31:00
  • Numpy数组转置的两种实现方法

    2023-01-22 16:36:54
  • pycharm不以pytest方式运行,想要切换回普通模式运行的操作

    2022-02-05 15:27:10
  • Git工作流模式及命令的使用讲解

    2023-09-17 08:31:28
  • MySQL中把varchar类型转为date类型方法详解

    2024-01-27 03:15:01
  • 导航与搜索合并的可能性

    2009-09-27 12:06:00
  • 分享10个免费超棒的编程用等宽字体

    2023-12-06 11:28:32
  • pandas使用fillna函数填充NaN值的代码实例

    2023-09-29 05:51:48
  • asp之家 网络编程 m.aspxhome.com