切片的基础知识

news2024/11/15 19:37:32

在这里插入图片描述

文章目录

    • ● Slice 的底层实现原理?
    • ● array 和 Slice 的区别?
    • ● 拷贝大切片一定比小切片代价大吗?
    • ● Slice 深拷贝和浅拷贝?
    • ● 零切片、空切片、nil切片?
    • ● Slice 的扩容机制?
    • ● Slice 为什么不是线程安全的?如何解决这个问题?
    • ● 参数传递切片和切片指针有什么区别?
    • ● range 遍历切片的时候有什么要注意的?

● Slice 的底层实现原理?

切片是对底层数组的引用,在创建一个切片的时候,Go会自动分配一个底层数组,并维护这个数组的指针、切片的长度和容量。

● array 和 Slice 的区别?

1、数组是固定长度的,不能动态扩容,在编译期就会确定大小,切片是可以自动扩容的。 2、数组是值类型,切片是引用类型,每个切片都引用了一个底层数组,切片本身不能存储任何数据,都是底层数组存储数据,所以修改切片的时候修改的是底层数组中的数据。切片一旦扩容,指向一个新的底层数组,内存地址也就随之改变。

● 拷贝大切片一定比小切片代价大吗?

不一定。切片中的第一个字段是指向切片底层数组的指针,这是切片的存储空间,第二个字段是切片的长度,第三个字段是容量。将一个切片变量分配给另一个变量只会复制三个机器字,大切片跟小切片的区别无非就是 Len 和 Cap的值比小切片的这两个值大一些,如果发生拷贝,本质上就是拷贝上面的三个字段。

● Slice 深拷贝和浅拷贝?

浅拷贝是创建一个新的切片,但这个新切片仍然引用原切片的底层数组。 深拷贝是创建一个新的切片,并复制原来切片的所有元素到新的切片中。

● 零切片、空切片、nil切片?

零切片
我们把切片内部数组的元素都是零值或者底层数组的内容就全是 nil的切片叫做零切片,使用make创建的、长度、容量都不为0的切片就是零值切片:slice := make([]int,5) // 0 0 0 0 0 slice := make([]*int,5) // nil nil nil nil nil
nil切片
nil切片的长度和容量都为0,并且和nil比较的结果为true,采用直接创建切片的方式或new创建切片的方式:var slice []int var slice = *new([]int)
空切片
空切片的长度和容量也都为0,但是和nil的比较结果为false,因为所有的空切片的数据指针都指向同一个地址 0xc42003bda0;使用字面量、make可以创建空切片:var slice = []int{} var slice = make([]int, 0) 05. 切片的扩容策略

● Slice 的扩容机制?

新申请的容量如果大于当前容量的两倍,会将新申请的容量直接作为新的容量,如果新申请的容量小于当前容量的两倍,会有一个阈值,即当前切片容量小于256时,切片会将当前容量的2倍作为新申请的容量,如果大于等于256,每次容量会增加 (旧容量+3*256)/4。

● Slice 为什么不是线程安全的?如何解决这个问题?

切片的底层数据结构体包括一个指向数组的指针,在多个线程同时对 切片进行操作时会导致竞态条件。(并发读写问题、切片扩容问题) 通过使用适当的同步机制如互斥锁、读写锁,确保对切片的并发访问是安全的,从而避免竞态条件和数据破坏。 (竞态条件是指在并发程序中,由于多个线程或goroutine同时访问或修改共享资源,从而导致程序的执行结果依赖于线程或goroutine的调度顺序,可能产生不确定和错误的结果。)

● 参数传递切片和切片指针有什么区别?

当切片作为参数传递时,其实就是一个结构体的传递,因为Go语言参数传递只有值传递,传递一个切片就会浅拷贝原切片,但因为底层数据的地址没有变,所以在函数内对切片的修改,也将会影响到函数外的切片。
在 Go 语言中,切片虽然是引用类型,但在函数中使用 append 操作时,需要注意一些细节。append 函数可能会导致切片底层数组重新分配内存,从而影响切片的引用传递。
具体来说,当 append 导致切片容量不足而需要扩容时,append 操作会创建一个新的底层数组,并返回新的切片引用。
由于你没有将这个新引用返回给原切片,原切片仍然指向旧的底层数组,因此不会体现出引用传递的效果。
参数传递切片指针就很容易理解了,如果你想修改切片中元素的值,并且更改切片的容量和底层数组,则应该按指针传递。

● range 遍历切片的时候有什么要注意的?

副本与原始切片: 在range遍历中,实际上会创建每个元素的副本。这意味着你在循环中对副本的修改不会影响原始切片。如果需要修改原始切片,应该使用索引来操作。
只读: 默认情况下,range遍历是只读的,不能修改切片的元素。如果尝试在range循环中修改元素值,会引发编译错误。
索引和值的顺序: 在range循环中,索引总是在前,元素值总是在后。如果只需要索引而不需要元素值,可以使用下划线 _ 来忽略元素值。
切片遍历的性能考虑:尽管使用range遍历切片是非常方便的,但在性能方面可能会有一些影响。每次使用range遍历时,都会创建元素的副本,这可能导致额外的内存开销和性能下降。如果在性能敏感的场景下遇到性能问题,你可以考虑使用索引遍历,以避免创建副本。另外,如果需要在遍历时修改切片的元素,也应该使用索引遍历,以确保修改能够直接影响到原始切片。
切片为空: 如果切片为空,使用range遍历不会执行任何循环体,这可能会导致你的代码出现逻辑问题。在使用range之前,最好先检查切片的长度。
切片为nil: 如果切片为nil,使用range遍历会引发运行时错误。同样,遍历之前应该确保切片不为nil。
遍历数组 vs. 切片: 在数组上使用range遍历和在切片上使用range遍历是不同的。对于数组,range返回索引和元素的副本,而在切片上,range返回索引和元素值的副本。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1883550.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

HarmonyOS Next开发学习手册——Native XComponent

场景介绍 Native XComponent是XComponent组件提供在Native层的实例,可作为JS层和Native层XComponent绑定的桥梁。XComponent所提供的NDK接口都依赖于该实例。接口能力包括获取Native Window实例、获取XComponent的布局/事件信息、注册XComponent的生命周期回调、注…

理性决策的艺术:从购房到择偶的数学智慧;37% 规则,做出最佳决策的秘诀;用数学模型解决人生难题

在面对人生重大决策时,如购房或择偶,我们常常感到迷茫和困惑。然而,如果我们能够将这些看似复杂的问题简化为数学模型,我们就能以更加理性和系统的方式做出决策。 37%规则 1950年代,当时几位数学家开始研究这样一个问…

钉钉开放AI生态战略的真正价值到底是什么?很多人都没看懂

来源: 首席数智官 hello 大家好,我们是数字化领军者都在看的首席数智官。 关注我,每天给你讲一个商业案例。 今天我们要给你讲的是:钉钉开放AI大模型生态的战略意义到底是什么? 「谁先赢得苹果,谁就赢得…

C++实现简化版Qt的QObject(3):增加父子关系、属性系统

前几天写了文章: C实现一个简单的Qt信号槽机制 C实现简化版Qt信号槽机制(2):增加内存安全保障 之后感觉还不够过瘾,Qt中的QObject体系里还有不少功能特性没有实现。为了提高QObject的还原度,今天我们将父子…

欢迎回家!揭秘“嫦娥六号”背后的守望者

6月25日,嫦娥六号返回器携带来自月背的月球样品安全着陆在内蒙古四子王旗预定区域。这是时隔3年多后,中国探月工程的又一关键节点任务,也是时隔5年多后,嫦娥探测器再去月球背面。 在此次任务中,同元软控数字伴飞团队为…

C++使用Poco库封装一个HTTP客户端类

0x00 前言 我们在使用HTTP协议获取接口数据时,通常需要在Header和Query中添加参数,还有一种就是在Body中追加XML或者JSON格式的数据。本文主要讲述使用Poco库提交HTTP Post请求的Body中附加XML格式的数据,JSON格式的数据类似。 0x01 HttpCl…

Zuul介绍

Zuul 是 Netflix 开源的一个云平台网络层代理,它主要用于路由、负载均衡、中间件通信和动态路由。Zuul 本质上是一个基于 JVM 的网关,它提供了以下功能: 1.路由:Zuul 允许客户端和服务器之间的所有入站和出站请求通过一个中心化的…

Hadoop页面报错Permission denied: user=dr.who, access....

1、临时解决 hdfs dfs -chmod -R 777 /这种方法,存在一个不足,就是后面重新创建的文件夹,页面进行删除的时候,依然报这个错。 但是,对于应付紧急客户需求,可以临时用一下。 2、永久解决 查看页面的Owner…

超声波清洗机怎么选?极力推荐四款口碑大牌超声波清洗机

相信大家都知道超声波清洗机,每次眼镜脏的时候,去眼镜店里让老板帮忙清洗,她们用的就是超声波清洗机,通过超声波的原理深入物品深处清洁,清洁效果非常好。相对手洗的方式,超声波清洗机能够保护镜片在清洗过…

SpringCloud_Ribbon负载均衡

概述 SpringCloud底层其实是利用了一个名为Ribbon的组件,来实现负载均衡功能的。 源码 LoadBalancerInterceptor 其中含有intercept方法,拦截用户的HttpRequest请求: request.getURI() 获取请求uri,即http://userservice/use…

【哈哈大一上学的全忘了,重开!!】STM32从零入门物联网开发

本笔记资料来源 :STM32物联网入门30步=单片机物联网入门教程 WIFI连接阿里云物联网CubeMXHAL库蓝牙ESP8266杜洋主讲_哔哩哔哩_bilibili IOT:Internet of things 学习目标: 1.掌握洋桃IoT开发板的各功能以及驱动与基本应用 2.掌…

Python容器 之 字符串--字符串的常用操作方法

1.字符串查找方法 find() 说明:被查找字符是否存在于当前字符串中。 格式:字符串.find(被查找字符) 结果:如果存在则返回第一次出现 被查找字符位置的下标 如果不存在则返回 -1 需求: 1. 现有字符串数据: 我是中国人 2. 请设计程序…

vue根据文字长短展示跑马灯效果

介绍 为大家介绍一个我编写的vue组件 auto-marquee ,他可以根据要展示文本是否超出展示区域,来判断是否使用跑马灯效果,效果图如下所示 假设要展示区域的宽度为500px,当要展示文本的长度小于500px时,只会展示文本&…

mysql5.7.30忘记root密码

windows系统安装了mysql5.7.30,在使用navicat链接mysql时候,提示 如何解决: 打开任务管理器的服务,查看有没有MYSQL服务。 如果没有,则按照下面的csdn博客进行操作。 https://blog.csdn.net/clj198606061111/article…

PostgreSQL的学习心得和知识总结(一百四十七)|深入理解PostgreSQL数据库之transaction chain的使用和实现

目录结构 注:提前言明 本文借鉴了以下博主、书籍或网站的内容,其列表如下: 1、参考书籍:《PostgreSQL数据库内核分析》 2、参考书籍:《数据库事务处理的艺术:事务管理与并发控制》 3、PostgreSQL数据库仓库…

Firewalld 概述

1.firewalld简介 firewalld的作用是为包过滤机制提供匹配规则(或称为策略),通过各种不同的规则,告诉 netfiter对来自指定源、前往指定目的或具有某些协议特征的数据包采取何种处理方式。 为了更加方便地组织和管理防火墙,firewalld 提供了支持网络区域…

【笔记】手工部署之linux中开放已安装的mysql与tomcat端口

在需要打包的springboot项目中输入mvn clean package 在target下面获得jar包 进入linux中你想要该jar包存在的位置 将jar包上传至linux中 此时在浏览器中输入linux的ip地址:端口号/mapping路径为404 故: 在linux中另开一个标签页 检查mysql和tomcat已…

JMH319【亲测整理】2017剑侠情缘2剑网2汉化版+网游VM单机版+修复纹饰翅膀+内置GM命令无限道具+一键端视频安装教学

资源介绍: 这一套新端早就在手上 一直没分享出来 既然大家放出来了 我也就发出来大家研究吧 目前在改另外一套 是否需要虚拟机:是 文件大小:压缩包约7G 支持系统:win7、win10、win11 硬件需求:运行内存12G …

遇到多语言跨境电商系统源码问题?这里有解决方案!

从手机到电脑,从线下到线上,如今,跨境电商正在打破地域界限,成为全球贸易的新引擎。在这个全球化的背景下,跨境电商平台的运营也面临着一系列的挑战,其中之一就是多语言问题。如果你遇到了多语言跨境电商系…

c进阶篇(三):字符串函数

1.strlen: strlen - C Reference strlen 函数是一个标准库函数&#xff0c;用于计算以 null 结尾的字符串的长度&#xff0c;也就是字符串中实际字符的数量&#xff0c;不包括最后的 null 终止符 \0。它定义在 <string.h> 头文件中。 函数原型:size_t strlen(const ch…