深入剖析Go語(yǔ)言中數(shù)組和切片的區(qū)別
在 Go 語(yǔ)言中,數(shù)組和切片是兩個(gè)常用的數(shù)據(jù)結(jié)構(gòu)。它們都可以用于存儲(chǔ)一組相同類型的元素,但在底層實(shí)現(xiàn)和使用方式上存在一些重要的區(qū)別。本文將深入探討 Go 語(yǔ)言數(shù)組和切片的區(qū)別,包括它們的定義、內(nèi)存布局、長(zhǎng)度和容量、初始化和操作等方面。通過(guò)全面了解這兩種數(shù)據(jù)結(jié)構(gòu)的特性,能夠更好地在實(shí)際開(kāi)發(fā)中選擇和使用合適的數(shù)據(jù)結(jié)構(gòu),提高代碼的效率和可維護(hù)性。
1. 數(shù)組的定義和特性
1.1 數(shù)組的定義
數(shù)組是一種固定長(zhǎng)度的數(shù)據(jù)結(jié)構(gòu),用于存儲(chǔ)具有相同類型的元素。在 Go 語(yǔ)言中,數(shù)組的定義形式為:
var arrayName [length]elementType
其中,arrayName 是數(shù)組的名稱,length 是數(shù)組的長(zhǎng)度,elementType 是數(shù)組元素的類型。例如,下面是一個(gè)包含 5 個(gè)整數(shù)的數(shù)組的定義:
var numbers [5]int
1.2 數(shù)組的特性
- 數(shù)組的長(zhǎng)度在定義后是不可更改的,因此它是一個(gè)固定大小的容器。
- 數(shù)組的內(nèi)存布局是連續(xù)的,元素按照其定義的順序依次存儲(chǔ)。
- 可以通過(guò)索引訪問(wèn)和修改數(shù)組中的元素。
2. 切片的定義和特性
2.1 切片的定義
切片是一個(gè)可變長(zhǎng)度的序列,它是基于數(shù)組的抽象。在 Go 語(yǔ)言中,切片的定義形式為:
var sliceName []elementType
其中,sliceName 是切片的名稱,elementType 是切片元素的類型。切片沒(méi)有固定的長(zhǎng)度,可以根據(jù)需要?jiǎng)討B(tài)增加或減少。切片底層依賴數(shù)組,但與數(shù)組相比,切片提供了更靈活和方便的操作方法。
2.2 切片的特性
- 切片的長(zhǎng)度表示切片當(dāng)前包含的元素個(gè)數(shù)。
- 切片的容量表示切片底層數(shù)組的大小,即切片可以訪問(wèn)的元素個(gè)數(shù)。
- 切片的內(nèi)存布局包含了一個(gè)指向底層數(shù)組的指針、長(zhǎng)度和容量信息。
- 可以通過(guò)索引訪問(wèn)和修改切片中的元素。
- 切片提供了方便的添加和刪除元素的方法。
3. 數(shù)組和切片的內(nèi)存布局
3.1 數(shù)組的內(nèi)存布局
數(shù)組是一段連續(xù)的內(nèi)存塊,元素依次排列在內(nèi)存中。例如,對(duì)于一個(gè)包含 5 個(gè)整數(shù)的數(shù)組,它們?cè)趦?nèi)存中的布局是這樣的:
在內(nèi)存中,數(shù)組的元素按照其定義的順序依次存儲(chǔ),每個(gè)元素占據(jù)相同大小的內(nèi)存空間。數(shù)組是一個(gè)由固定長(zhǎng)度的特定類型元素組成的序列,一個(gè)數(shù)組可以由零個(gè)或多個(gè)元素組成。數(shù)組的長(zhǎng)度是數(shù)組類型的組成部分。因?yàn)閿?shù)組的長(zhǎng)度是數(shù)組類型的組成部分,不同長(zhǎng)度或不同類型的數(shù)據(jù)組成的數(shù)組都是不同的類型。例如:[5]int 和 [6]int 是不同類型的數(shù)組,[5]string 和 [5]int 也是不同類型的數(shù)組。
3.2 切片的內(nèi)存布局
簡(jiǎn)單地說(shuō),切片(slice)就是一種簡(jiǎn)化版的動(dòng)態(tài)數(shù)組。因?yàn)閯?dòng)態(tài)數(shù)組的長(zhǎng)度不固定,所以切片的長(zhǎng)度自然也就不能是類型的組成部分了。數(shù)組雖然有適用的地方,但是數(shù)組的類型和操作都不夠靈活,因此在 Go 代碼中數(shù)組使用的并不是很多,而切片則使用的相當(dāng)廣泛。
我們看看切片的結(jié)構(gòu)定義,即 reflect.SliceHeader:
type SliceHeader struce { Data uintptr Len int Cap int }
切片是一種引用類型,它有三個(gè)屬性:
- 指針(Pointer):指向底層數(shù)組的起始位置的指針。
- 長(zhǎng)度(Length):切片當(dāng)前包含的元素個(gè)數(shù)。
- 容量(Capacity):切片底層數(shù)組的大小,即切片可以訪問(wèn)的元素個(gè)數(shù)。
通過(guò)這些信息,切片可以動(dòng)態(tài)調(diào)整自己的長(zhǎng)度,并訪問(wèn)底層數(shù)組中的元素。需要注意的是,切片的容量可以大于長(zhǎng)度,表示切片中剩余的可用空間。
下圖展示了對(duì)于一個(gè)包含 5 個(gè)整數(shù)的切片,它們?cè)趦?nèi)存中的布局是這樣的:
4. 數(shù)組和切片的長(zhǎng)度和容量
4.1 數(shù)組的長(zhǎng)度和容量
數(shù)組的長(zhǎng)度在定義時(shí)就確定了,無(wú)法更改。我們可以使用內(nèi)置的 len 函數(shù)獲取數(shù)組的長(zhǎng)度。例如:
var numbers [5]int length := len(numbers) fmt.Println(length) // 輸出:5
由于數(shù)組的長(zhǎng)度是固定的,所以它沒(méi)有容量的概念。每個(gè)數(shù)組都有確切的大小,無(wú)法在運(yùn)行時(shí)動(dòng)態(tài)改變。
4.2 切片的長(zhǎng)度和容量
切片具有動(dòng)態(tài)調(diào)整長(zhǎng)度的能力。在創(chuàng)建切片時(shí),可以指定切片的長(zhǎng)度和容量,也可以根據(jù)已有的數(shù)組或切片創(chuàng)建切片。
切片的長(zhǎng)度表示切片當(dāng)前包含的元素個(gè)數(shù),可以通過(guò)內(nèi)置的 len 函數(shù)獲取。切片的容量表示切片底層數(shù)組的大小,可以通過(guò)內(nèi)置的 cap 函數(shù)獲取。
array := [5]int{1, 2, 3, 4, 5} slice := array[1:3] // 創(chuàng)建一個(gè)切片,包含數(shù)組中索引為1和2的元素 length := len(slice) capacity := cap(slice) fmt.Println(length) // 輸出:2 fmt.Println(capacity) // 輸出:4
在上面的例子中,切片 slice 的長(zhǎng)度為2,因?yàn)樗藬?shù)組中索引為1和2的兩個(gè)元素。切片的容量為4,因?yàn)樗讓訑?shù)組的大小為5,從索引1開(kāi)始的剩余空間為4。
當(dāng)切片的長(zhǎng)度超過(guò)容量時(shí),切片會(huì)自動(dòng)擴(kuò)容以適應(yīng)新的元素。這是由 Go 語(yǔ)言的運(yùn)行時(shí)系統(tǒng)自動(dòng)處理的,開(kāi)發(fā)者無(wú)需手動(dòng)管理內(nèi)存。當(dāng)切片擴(kuò)容時(shí),系統(tǒng)會(huì)創(chuàng)建一個(gè)新的更大的底層數(shù)組,并將舊數(shù)組中的元素復(fù)制到新數(shù)組中。
5. 數(shù)組和切片的初始化
5.1 數(shù)組的初始化
數(shù)組的初始化可以通過(guò)以下幾種方式進(jìn)行:
使用數(shù)組字面量(Array Literal)初始化數(shù)組的每個(gè)元素。
var numbers = [5]int{1, 2, 3, 4, 5}
根據(jù)索引初始化數(shù)組的特定元素。
var numbers [5]int numbers[0] = 1 numbers[1] = 2
5.2 切片的初始化
切片的初始化可以通過(guò)以下幾種方式進(jìn)行:
使用切片字面量(Slice Literal)初始化切片。
slice := []int{1, 2, 3, 4, 5}
基于已有的數(shù)組或切片創(chuàng)建切片。
array := [5]int{1, 2, 3, 4, 5} slice := array[1:3] // 創(chuàng)建一個(gè)切片,包含數(shù)組中索引為1和2的元素
使用內(nèi)置的 make 函數(shù)創(chuàng)建指定長(zhǎng)度和容量的切片。
slice := make([]int, 5) // 創(chuàng)建一個(gè)長(zhǎng)度為5的切片,初始值為0 slice := make([]int, 5, 10) // 創(chuàng)建一個(gè)長(zhǎng)度為5、容量為10的切片
6. 數(shù)組和切片的操作
6.1 訪問(wèn)元素
數(shù)組和切片都可以通過(guò)索引來(lái)訪問(wèn)其中的元素。索引從0開(kāi)始,范圍是0到長(zhǎng)度減1。以下是訪問(wèn)數(shù)組和切片元素的示例:
numbers := [5]int{1, 2, 3, 4, 5} fmt.Println(numbers[0]) // 輸出:1 ? slice := numbers[1:3] fmt.Println(slice[1]) // 輸出:3
6.2 修改元素
數(shù)組和切片的元素都可以被修改。通過(guò)索引將新值賦給相應(yīng)的元素即可。以下是修改數(shù)組和切片元素的示例:
numbers := [5]int{1, 2, 3, 4, 5} numbers[2] = 10 // 修改數(shù)組中索引為2的元素為10 fmt.Println(numbers) // 輸出:[1 2 10 4 5] ? slice := numbers[1:3] slice[0] = 20 // 修改切片中索引為0的元素為20 fmt.Println(numbers) // 輸出:[1 20 10 4 5]
6.3 添加元素
由于數(shù)組的長(zhǎng)度是固定的,無(wú)法直接添加新元素。但可以通過(guò)創(chuàng)建一個(gè)新的更大的數(shù)組,并將舊數(shù)組中的元素復(fù)制到新數(shù)組來(lái)實(shí)現(xiàn)類似添加元素的效果。
切片則提供了更便捷的方法來(lái)添加元素。使用內(nèi)置的 append 函數(shù)可以向切片末尾添加一個(gè)或多個(gè)元素。append 函數(shù)會(huì)自動(dòng)處理切片的擴(kuò)容和元素的復(fù)制。以下是向切片添加元素的示例:
slice := []int{1, 2, 3} slice = append(slice, 4) // 向切片末尾添加元素4 fmt.Println(slice) // 輸出:[1 2 3 4] ? slice = append(slice, 5, 6, 7) // 向切片末尾添加多個(gè)元素 fmt.Println(slice) // 輸出:[1 2 3 4 5 6 7]
6.4 刪除元素
數(shù)組無(wú)法直接刪除元素,但可以通過(guò)重新賦值的方式間接刪除元素。切片則提供了更方便的方法來(lái)刪除元素。
使用切片的切片操作可以刪除指定位置的元素。以下是刪除切片中指定元素的示例:
slice := []int{1, 2, 3, 4, 5} slice = append(slice[:2], slice[3:]...) // 刪除切片中索引為2的元素 fmt.Println(slice) // 輸出:[1 2 4 5]
通過(guò)將索引為2之前的元素和索引為2之后的元素重新拼接在一起,即可刪除索引為2的元素。
7. 總結(jié)
通過(guò)本文的講解,我們深入理解了 Go 語(yǔ)言數(shù)組和切片的區(qū)別。數(shù)組是固定長(zhǎng)度的數(shù)據(jù)結(jié)構(gòu),長(zhǎng)度不可更改,而切片是可變長(zhǎng)度的序列,基于數(shù)組的抽象。數(shù)組的內(nèi)存布局是連續(xù)的,而切片包含了指向底層數(shù)組的指針、長(zhǎng)度和容量信息。數(shù)組的操作受限,無(wú)法直接添加或刪除元素,而切片提供了方便的添加和刪除元素的方法。切片的擴(kuò)容由運(yùn)行時(shí)系統(tǒng)自動(dòng)處理,無(wú)需手動(dòng)管理內(nèi)存。
通過(guò)深入理解數(shù)組和切片的區(qū)別,我們能夠更好地選擇和使用合適的數(shù)據(jù)結(jié)構(gòu),提高代碼的效率和可維護(hù)性。同時(shí),我們也掌握了數(shù)組和切片的初始化、訪問(wèn)、修改、添加和刪除等基本操作。
到此這篇關(guān)于深入剖析Go語(yǔ)言中數(shù)組和切片的區(qū)別的文章就介紹到這了,更多相關(guān)Go語(yǔ)言數(shù)組 切片內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- go中實(shí)現(xiàn)字符切片和字符串互轉(zhuǎn)
- Golang切片連接成字符串的實(shí)現(xiàn)示例
- Go語(yǔ)言之重要數(shù)組類型切片(slice)make,append函數(shù)解讀
- GO語(yǔ)言中創(chuàng)建切片的三種實(shí)現(xiàn)方式
- golang字符串切片去重的幾種算法
- 詳解golang的切片擴(kuò)容機(jī)制
- 一文詳解Go語(yǔ)言中切片的底層原理
- Go?語(yǔ)言中切片的三種特殊狀態(tài)
- 淺談go中切片比數(shù)組好用在哪
- 一文總結(jié)Go語(yǔ)言切片核心知識(shí)點(diǎn)和坑
- Go語(yǔ)言實(shí)戰(zhàn)之切片內(nèi)存優(yōu)化
- Go切片的具體使用
相關(guān)文章
Golang?channel為什么不會(huì)阻塞的原因詳解
這篇文章主要為大家介紹了Golang?channel為什么不會(huì)阻塞的原因詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-07-07Go語(yǔ)言基礎(chǔ)學(xué)習(xí)之?dāng)?shù)組的使用詳解
數(shù)組相必大家都很熟悉,各大語(yǔ)言也都有數(shù)組的身影。Go 語(yǔ)言也提供了數(shù)組類型的數(shù)據(jù)結(jié)構(gòu)。本文就來(lái)通過(guò)一些簡(jiǎn)單的示例帶大家了解一下Go語(yǔ)言中數(shù)組的使用,希望對(duì)大家有所幫助2022-12-12基于Go+OpenCV實(shí)現(xiàn)人臉識(shí)別功能的詳細(xì)示例
OpenCV是一個(gè)強(qiáng)大的計(jì)算機(jī)視覺(jué)庫(kù),提供了豐富的圖像處理和計(jì)算機(jī)視覺(jué)算法,本文將向你介紹在Mac上安裝OpenCV的步驟,并演示如何使用Go的OpenCV綁定庫(kù)進(jìn)行人臉識(shí)別,需要的朋友可以參考下2023-07-07Golang中 import cycle not allowed 問(wèn)題
這篇文章主要介紹了Golang中 import cycle not allowed 問(wèn)題的解決方法,問(wèn)題從描述到解決都非常詳細(xì),需要的小伙伴可以參考一下2022-03-03Go語(yǔ)言實(shí)現(xiàn)登錄驗(yàn)證代碼案例
這篇文章主要介紹了Go語(yǔ)言實(shí)現(xiàn)登錄驗(yàn)證代碼案例,代碼和圖文講解的很清晰,有感興趣的可以學(xué)習(xí)下2021-03-03一文帶你了解Golang中select的實(shí)現(xiàn)原理
select是go提供的一種跟并發(fā)相關(guān)的語(yǔ)法,非常有用。本文將介紹?Go?語(yǔ)言中的?select?的實(shí)現(xiàn)原理,包括?select?的結(jié)構(gòu)和常見(jiàn)問(wèn)題、編譯期間的多種優(yōu)化以及運(yùn)行時(shí)的執(zhí)行過(guò)程2023-02-02