Go 切片:用法和本质

要想更好的了解一个知识点,实战是最好的经历。

题目

我这里放一道题目:

Go 复制代码
package main

import "fmt"

func SliceRise(s []int) {
    s = append(s, 0)
    for i := range s {
       s[i]++
    }
    fmt.Println(s)
}

func SlicePrint() {
    s1 := []int{1, 2}
    s2 := s1
    s2 = append(s2, 3)
    SliceRise(s1)
    SliceRise(s2)
    fmt.Println(s1, s2)
}

func main() {
    SlicePrint()
}

大家猜猜本题的运行结果是多少?

s1: 1、2 **|| s2:**2、3、4

知识点

是不是蒙了( •̀ ω •́ )✧,蒙了才有效果,才能更加明白自己的不足。

以上涉及了好几个知识点。我一一讲起。

1、切片的底层数据结构

Go 复制代码
type Slice struct{
    array   unsafe.Pointer // 储存着数组存放地址
    len     int            // 切片长度
    cap     int            // 切片容量
}

切片的底层数据结构是一个 "结构体"

2、切片的扩容机制

slice的扩容遵循以下原则

1、如果slice容量小于1024,则新slice容量扩大为原来的2倍。

2、如果slice容量大于等于1024,采用1.25倍,则新的slice容量将扩大为原来的1.25倍

这里有一点需要注意,当slice扩容时,会开辟一个新的空间,将旧的slice依次复制进入。


学到这里,其实就已经基本OK了,接下来我会放出本题的解析。

解析

大家都知道,切片是引用传递。什么是引用传递呢?就是传递地址值。

但是有一点需要注意,切片的数据结构是结构体,底层是struct。

所以传递时,将会创造一个新的结构体 ,把老结构体全部粘贴复制进去(地址,len、cap)

s1 := s2

如上,就相当于

相当于把两个结构体,直接复制并拷贝过去复制了一个一模一样的。

显而易见,这两个,是不同的结构体。--> 但是却共同一个共同的地址0111,也就是同一个数组。

到这里大家应该就明白了,他们传递时,传递的是地址。结构体是直接复制过去的一个新的。

s = append(s,0)时,由于新添加了一个值,导致容量扩大,cap增加。这时就会分配一个新地址给s结构体。而原来的s1结构体内,存的地址不会改变。

故fmt.Println(s1)是:1、2。

而s2,用同样的分析,亦可得出结论。

相关推荐
贵州晓智信息科技4 分钟前
Three.js实现动态水泡效果逐步解析GLSL着色器
开发语言·javascript·着色器
汤姆和杰瑞在瑞士吃糯米粑粑20 分钟前
【优先算法】滑动窗口--结合例题详解学习
开发语言·数据结构·c++·算法
code_shenbing1 小时前
C# 控制打印机:从入门到实践
开发语言·c#
magic 2451 小时前
JVM体系结构
java·开发语言·jvm·intellij-idea·idea
计算机-秋大田1 小时前
基于微信小程序的购物系统设计与实现(LW+源码+讲解)
java·后端·微信小程序·小程序·课程设计
code_shenbing1 小时前
C# 解析 HTML 实战指南
开发语言·c#·html
ekskef_sef1 小时前
在2023idea中如何创建SpringBoot
java·spring boot·后端
Sao_E2 小时前
SpringBoot实现定时任务,使用自带的定时任务以及调度框架quartz的配置使用
java·spring boot·后端
蟹至之2 小时前
类和对象(3)——继承:extends关键字、super关键字、protected关键字、final关键字
java·开发语言·继承·类和对象
万亿少女的梦1682 小时前
基于PHP的校园兼职系统的设计与开发
开发语言·网络·数据库·爬虫·网络安全·php