網站首頁 編程語言 正文
定義
切片區別于數組,是引用類型, 不是值類型。數組是固定長度的,而切片長度是可變的,我的理解是:切片是對數組一個片段的引用。
var s1 []int //定義一個存放int類型元素的切片 var s2 []string //定義一個存放string類型元素的切片 fmt.Println(s1, s2) fmt.Println(s1 == nil) //true 為空 沒有開辟內存空間 fmt.Println(s2 == nil) //true
打印結果:
解析: 說明我們已經聲明定義成功了,但是并沒有開辟內存空間,因為s1、s2的值為nil
定義并初始化
我們可以在定義的同時初始化
var s1 = []int{1, 2, 3} var s2 = []string{"北苑", "長陽", "望京"} fmt.Println(s1, s2) fmt.Println(s1 == nil) //false fmt.Println(s2 == nil) //false
打印結果:
解析: 初始化成功,s1 s2的值都不等于nil
長度和容量
分別使用len()、cap()獲得切片的長度和容量
fmt.Printf("len(s1):%d cap(s1):%d\n", len(s1), cap(s1)) fmt.Printf("len(s2):%d cap(s2):%d\n", len(s2), cap(s2))
打印結果:
解析: 和我們預期的一致,長度和容量都為3
由數組得到切片
開篇我已經提到數組和切片的關系,這里在進一步講一下:
- 切片的本質是操作數組,只是數組是固定長度的,而切片的長度可變的
- 切片是引用類型,可以理解為引用數組的一個片段;而數組是值類型,把數組A賦值給數組B,會為數組B開辟新的內存空間,修改數組B的值并不會影響數組A。而切片作為引用類型,指向同一個內存地址,是會互相影響的。
//定義一個數組 a1 := [...]int{1, 2, 3, 4, 5, 6, 7, 8, 9} s3 := a1[0:4] //基于一個數組切割 [0:4]左包含 右不包含 即為[1,2,3,4] fmt.Println(s3)
打印結果:
注意:a1[0:4] 基于一個數組切割 [0:4]左包含 右不包含 即為[1,2,3,4]
更多切割方式舉例
a1 := [...]int{1, 2, 3, 4, 5, 6, 7, 8, 9} s4 := a1[2:4] //[3 4] s5 := a1[:4] //[1 2 3 4] s6 := a1[2:] //[3 4 5 6 7 8 9] s7 := a1[:] //[1 2 3 4 5 6 7 8 9] fmt.Println(s4) fmt.Println(s5) fmt.Println(s6) fmt.Println(s7)
打印結果:
解析: 都符合上面提到的左包含,右不包含
原則 s4從下標2開始截取,截取到下標4 s5省略了第一個參數,表示從下標0開始截取 s6省略了第二個參數,表示截取到最后一個元素 s7省略了兩個參數,只填寫了中間的冒號:,表示取全部元素
切片的長度和容量
切片的長度很好理解,就是元素的個數
切片的容量我們重點理解一下:在切片引用的底層數組中從切片的第一個元素到數組最后一個元素的長度(元素數量)
這么讀起來可能有點抽象,我們看下面這個栗子就很好理解啦:
a1 := [...]int{1, 2, 3, 4, 5, 6, 7, 8, 9} s5 := a1[:4] //[1 2 3 4] s6 := a1[2:] //[3 4 5 6 7 8 9] s7 := a1[:] //[1 2 3 4 5 6 7 8 9] fmt.Printf("len(s5):%d cap(s5):%d\n", len(s5), cap(s5)) //4 9 fmt.Printf("len(s6):%d cap(s6):%d\n", len(s6), cap(s6)) //7 7 fmt.Printf("len(s7):%d cap(s7):%d\n", len(s7), cap(s7)) //9 9
打印結果:
解析: a1是數組長度為9,容量也為9,值是從1~9
s5/s6/s7都是切割數組a1得到的切片。
s5的長度為4,因為只有1 2 3 4這4個元素,容量為9,因為s5切片的第一個元素是1,而s5底層數組a1最后一個元素是9,1~9共9個元素,所以s5的容量為9。
s6的長度為7,因為s6的元素是39這7個元素;容量也為7,因為s5的底層數組最后一個元素是9,39共7個元素,所以s6的容量為7。
S7更好理解了,長度和容量都是9,小伙伴們自己理解一下。
切片再切片
我們可以對切片進行再切片操作
比如,我們針對上面的數據再次切片進行測試
s8 :=s6[3:] fmt.Printf("len(s8):%d cap(s8):%d\n", len(s8), cap(s8)) //4 4
打印結果:
解析:我們知道可以對切片進行再次切片就可以,至于長度和容器大家搞明白上面的栗子,這個輸出結果就是意料之中的了。
slice是引用類型
我們舉個栗子來證明切片是引用類型
//定義數組 a1 := [...]int{1, 2, 3, 4, 5, 6, 7, 8, 9} //有數組切割成切片s6 s6 := a1[2:] //[3 4 5 6 7 8 9] //切片再次切片,賦值給s8 s8 :=s6[3:] //[6 7 8 9] //修改原始數組,把下標為2的值由3改為333 a1[2] = 333 //打印s6,發現s6中的3也變成了333 fmt.Println("s6:", s6) //[333 4 5 6 7 8 9] //因為s8基于s6切片而成,我們測試一下切片再切片的引用傳的 fmt.Println("s8:", s8) //[6 7 8 9] //我們把原始數組下標為5的值由6改為666 a1[5] = 666 //打印s8切片,得到結果6也變成了666 fmt.Println("s8:", s8) //[666 7 8 9]
打印結果:
解析: 由此我們可以明確的知道切片是引用類型,當底層數組改變時,不管是切片,還是切片再切片,值都會改變。因為他們使用的是一個內存塊,引用的一個內存地址。
總結
原文鏈接:https://juejin.cn/post/7071960543283642404
相關推薦
- 2022-10-03 Tomcat安裝使用及部署Web項目的3種方法匯總_Tomcat
- 2022-05-11 Feign之間的文件傳輸
- 2022-07-04 PyG搭建GCN模型實現節點分類GCNConv參數詳解_python
- 2023-01-30 Android?Https證書過期的兩種解決方案_Android
- 2022-06-15 C#數據類型實現背包、隊列和棧_C#教程
- 2022-04-16 python中的?sorted()函數和sort()方法區別_python
- 2022-12-04 深入了解Rust的切片使用_Rust語言
- 2022-04-24 解決redis在linux上的部署的問題_Redis
- 最近更新
-
- window11 系統安裝 yarn
- 超詳細win安裝深度學習環境2025年最新版(
- Linux 中運行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎操作-- 運算符,流程控制 Flo
- 1. Int 和Integer 的區別,Jav
- spring @retryable不生效的一種
- Spring Security之認證信息的處理
- Spring Security之認證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權
- redisson分布式鎖中waittime的設
- maven:解決release錯誤:Artif
- restTemplate使用總結
- Spring Security之安全異常處理
- MybatisPlus優雅實現加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務發現-Nac
- Spring Security之基于HttpR
- Redis 底層數據結構-簡單動態字符串(SD
- arthas操作spring被代理目標對象命令
- Spring中的單例模式應用詳解
- 聊聊消息隊列,發送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠程分支