亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

golang中的空slice案例

 更新時間:2021年04月26日 14:55:10   作者:晨夢思雨  
這篇文章主要介紹了golang中的空slice案例,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧

golang中允許對值為 nil 的 slice 添加元素

package main 
func main() {
 var s []int
 s = append(s, 1)
}

運行成功~

補充:golang slice 詳解

一、數(shù)組切片的使用

func main() {
	//1.基于數(shù)組創(chuàng)建數(shù)組切片
	var array [10]int = [10]int{1, 2, 3, 4, 5, 6, 7, 8, 9, 10}
	var slice = array[1:7] //array[startIndex:endIndex] 不包含endIndex
	//2.直接創(chuàng)建數(shù)組切片
	slice2 := make([]int, 5, 10)
	//3.直接創(chuàng)建并初始化數(shù)組切片
	slice3 := []int{1, 2, 3, 4, 5, 6}
	//4.基于數(shù)組切片創(chuàng)建數(shù)組切片
	slice5 := slice3[:4]
	//5.遍歷數(shù)組切片
	for i, v := range slice3 {
		fmt.Println(i, v)
	}
	//6.len()和cap()
	var len = len(slice2) //數(shù)組切片的長度
	var cap = cap(slice)  //數(shù)組切片的容量
	fmt.Println("len(slice2) =", len)
	fmt.Println("cap(slice) =", cap)
	//7.append() 會生成新的數(shù)組切片
	slice4 := append(slice2, 6, 7, 8)
	slice4 = append(slice4, slice3...)
	fmt.Println(slice4)
	//8.copy() 如果進行操作的兩個數(shù)組切片元素個數(shù)不一致,將會按照個數(shù)較小的數(shù)組切片進行復制
	copy(slice2, slice3) //將slice3的前五個元素復制給slice2
	fmt.Println(slice2, slice3)
}

二、數(shù)組切片數(shù)據結構分析

數(shù)組切片slice的數(shù)據結構如下,一個指向真實array地址的指針ptr,slice的長度len和容量cap

// slice 數(shù)據結構
type slice struct {
	array unsafe.Pointer 
	len   int            
	cap   int            
}

當傳參時,函數(shù)接收到的參數(shù)是數(shù)組切片的一個復制,雖然兩個是不同的變量,但是它們都有一個指向同一個地址空間的array指針,當修改一個數(shù)組切片時,另外一個也會改變,所以數(shù)組切片看起來是引用傳遞,其實是值傳遞。

三、append()方法解析

3.1 數(shù)組切片不擴容的情況

運行以下代碼思考一個問題:s1和s2是指向同一個底層數(shù)組嗎?

func main() {
	array := [20]int{1, 2, 3, 4, 5, 6, 7, 8, 9}
	s1 := array[:5]
	s2 := append(s1, 10)
	fmt.Println("s1 =", s1)
	fmt.Println("s2 =", s2)
	s2[0] = 0
	fmt.Println("s1 =", s1)
	fmt.Println("s2 =", s2)
}

輸出結果:

s1 = [1 2 3 4 5]

s2 = [1 2 3 4 5 10]

s1 = [0 2 3 4 5]

s2 = [0 2 3 4 5 10]

由第一行和第二行結果看來,似乎這是指向兩個不同的數(shù)組;但是當修改了s2,發(fā)現(xiàn)s1也跟著改變了,這又表明二者是指向同一個數(shù)組。到底真相是怎樣的呢?

運行以下代碼:

import (
	"fmt"
	"unsafe"
)
type Slice struct {
	ptr unsafe.Pointer // Array pointer
	len int            // slice length
	cap int            // slice capacity
}
func main() {
	array := [20]int{1, 2, 3, 4, 5, 6, 7, 8, 9}
	s1 := array[:5]
	s2 := append(s1, 10)
	s2[0] = 0
	// 把slice轉換成自定義的 Slice struct
	slice1 := (*Slice)(unsafe.Pointer(&s1))
	fmt.Printf("ptr:%v len:%v cap:%v \n", slice1.ptr, slice1.len, slice1.cap)
	slice2 := (*Slice)(unsafe.Pointer(&s2))
	fmt.Printf("ptr:%v len:%v cap:%v \n", slice2.ptr, slice2.len, slice2.cap)
}

輸出結果:

ptr:0xc04205e0a0 len:5 cap:20

ptr:0xc04205e0a0 len:6 cap:20

由結果可知:ptr指針存儲的是數(shù)組中的首地址的值,并且這兩個值相同,所以s1和s2確實是指向同一個底層數(shù)組。

但是,這兩個數(shù)組切片的元素不同,這個可以根據首地址和數(shù)組切片長度len來確定不同的數(shù)組切片應該包含哪些元素,因為s1和s2雖然指向同一個底層數(shù)組,但是二者的len不同。通過這個demo,也驗證了數(shù)組切片傳參方式也是值傳遞。

3.2 數(shù)組切片擴容的情況:

運行以下代碼,思考與不擴容情況的不同之處,以及為什么

func main() {
	s1 := []int{1, 2, 3, 4, 5, 6, 7, 8, 9}
	s2 := append(s1, 10)
	fmt.Println("s1 =", s1)
	fmt.Println("s2 =", s2)
	s2[0] = 0
	fmt.Println("s1 =", s1)
	fmt.Println("s2 =", s2)
}

輸出結果:

s1 = [1 2 3 4 5 6 7 8 9]

s2 = [1 2 3 4 5 6 7 8 9 10]

s1 = [1 2 3 4 5 6 7 8 9]

s2 = [0 2 3 4 5 6 7 8 9 10]

根據結果我們發(fā)現(xiàn),修改s2后,s1并未改變,這說明當append()后,s1和s2并未指向同一個底層數(shù)組,這又是為什么呢?

同樣,我們接著運行以下代碼:

import (
	"fmt"
	"unsafe"
)
type Slice struct {
	ptr unsafe.Pointer // Array pointer
	len int            // slice length
	cap int            // slice capacity
}
func main() {
	s1 := []int{1, 2, 3, 4, 5, 6, 7, 8, 9}
	s2 := append(s1, 10)
	fmt.Println("s1 =", s1)
	fmt.Println("s2 =", s2)
	s2[0] = 0
	fmt.Println("s1 =", s1)
	fmt.Println("s2 =", s2)
	// 把slice轉換成自定義的 Slice struct
	slice1 := (*Slice)(unsafe.Pointer(&s1))
	fmt.Printf("ptr:%v len:%v cap:%v \n", slice1.ptr, slice1.len, slice1.cap)
	slice2 := (*Slice)(unsafe.Pointer(&s2))
	fmt.Printf("ptr:%v len:%v cap:%v \n", slice2.ptr, slice2.len, slice2.cap)
}

輸出結果:

s1 = [1 2 3 4 5 6 7 8 9]

s2 = [1 2 3 4 5 6 7 8 9 10]

s1 = [1 2 3 4 5 6 7 8 9]

s2 = [0 2 3 4 5 6 7 8 9 10]

ptr:0xc04207a000 len:9 cap:9

ptr:0xc04207c000 len:10 cap:18

由結果可知:append()后,s1和s2確實指向了不同的底層數(shù)組,并且二者的數(shù)組容量cap也不相同了。

過程是這樣的:當append()時,發(fā)現(xiàn)數(shù)組容量不夠用,于是開辟了新的數(shù)組空間,cap變?yōu)樵瓉淼膬杀?,s2指向了這個新的數(shù)組,所以當修改s2時,s1不受影響

以上為個人經驗,希望能給大家一個參考,也希望大家多多支持腳本之家。如有錯誤或未考慮完全的地方,望不吝賜教。

相關文章

  • VSCode Golang dlv調試數(shù)據截斷問題及處理方法

    VSCode Golang dlv調試數(shù)據截斷問題及處理方法

    這篇文章主要介紹了VSCode Golang dlv調試數(shù)據截斷問題,本文通過實例代碼給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2020-06-06
  • Goland項目使用gomod配置的詳細步驟

    Goland項目使用gomod配置的詳細步驟

    Goland是一個用于Go語言開發(fā)的IDE,Goland的項目結構與Go語言的項目結構相似,下面這篇文章主要給大家介紹了關于Goland項目使用gomod配置的詳細步驟,文中通過圖文介紹的非常詳細,需要的朋友可以參考下
    2023-04-04
  • Go語言標準錯誤error全面解析

    Go語言標準錯誤error全面解析

    Go語言中的錯誤處理是通過內置的error接口來實現(xiàn)的,其中errorString和wrapError是兩種常見的錯誤類型實現(xiàn)方式,errorString通過errors.New()方法實現(xiàn),而wrapError則通過fmt.Errorf()方法實現(xiàn),支持錯誤的嵌套和解析
    2024-10-10
  • GO中Json解析的幾種方式

    GO中Json解析的幾種方式

    本文主要介紹了GO中Json解析的幾種方式,詳細的介紹了幾種方法,?文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2024-01-01
  • Gin與Mysql實現(xiàn)簡單Restful風格API實戰(zhàn)示例詳解

    Gin與Mysql實現(xiàn)簡單Restful風格API實戰(zhàn)示例詳解

    這篇文章主要為大家介紹了Gin與Mysql實現(xiàn)簡單Restful風格API示例詳解,有需要的朋友可以借鑒參考下希望能夠有所幫助,祝大家多多進步
    2021-11-11
  • Golang根據job數(shù)量動態(tài)控制每秒協(xié)程的最大創(chuàng)建數(shù)量方法詳解

    Golang根據job數(shù)量動態(tài)控制每秒協(xié)程的最大創(chuàng)建數(shù)量方法詳解

    這篇文章主要介紹了Golang根據job數(shù)量動態(tài)控制每秒協(xié)程的最大創(chuàng)建數(shù)量方法
    2024-01-01
  • GScript?編寫標準庫示例詳解

    GScript?編寫標準庫示例詳解

    這篇文章主要為大家介紹了GScript?編寫標準庫示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2022-10-10
  • Go泛型的理解和使用小結

    Go泛型的理解和使用小結

    泛型是一種非常強大的編程技術,可以提高代碼的復用性和可讀性,通過泛型容器和類型參數(shù)化,Go語言中的泛型可以實現(xiàn)更加靈活和通用的編程,提高代碼的復用性和可維護性,本文給大家介紹Go泛型的理解和使用,感興趣的朋友一起看看吧
    2023-12-12
  • Go語言中嵌入C語言的方法

    Go語言中嵌入C語言的方法

    這篇文章主要介紹了Go語言中嵌入C語言的方法,實例分析了Go語言中cgo工具的使用技巧,具有一定參考借鑒價值,需要的朋友可以參考下
    2015-02-02
  • Go語言處理Excel文件的教程詳解

    Go語言處理Excel文件的教程詳解

    在Go語言中,有許多庫和工具可用于處理Excel文件,本文將介紹如何使用Go語言處理Excel文件,包括讀取、寫入和修改Excel文件,需要的小伙伴可以了解下
    2024-01-01

最新評論