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

Go字典使用詳解

 更新時(shí)間:2022年11月18日 17:13:58   作者:Mingvvv  
今天和大家一起學(xué)習(xí)Go語言的字典。Go語言的字典又稱為map,一種使用廣泛的數(shù)據(jù)結(jié)構(gòu)。它是擁有key/value對(duì)元素的「無序集合」,而且在集合中key必須是唯一的

和許多編程語言一樣,在 Go 中,字典是一組鍵-值對(duì)( Go 中稱鍵-元素對(duì))的集合。

存儲(chǔ)/查找原理

當(dāng)我們要存儲(chǔ)或者查找某個(gè)鍵-元素對(duì)的時(shí)候,哈希表會(huì)先使用哈希函數(shù)將鍵值轉(zhuǎn)換為哈希值,哈希值一般是一個(gè)無符號(hào)的整數(shù)。

一個(gè)哈希表內(nèi)會(huì)存有一定數(shù)量的哈希桶,在字典的結(jié)構(gòu)里面,有一個(gè)屬性 B ,這個(gè)屬性代表當(dāng)前字典里面桶的個(gè)數(shù) (2^B) 。

	// A header for a Go map.
	type hmap struct {
	   // Note: the format of the hmap is also encoded in cmd/compile/internal/gc/reflect.go.
	   // Make sure this stays in sync with the compiler's definition.
	   count     int // # live cells == size of map.  Must be first (used by len() builtin)
	   flags     uint8
	   B         uint8  // log_2 of # of buckets (can hold up to loadFactor * 2^B items)
	   noverflow uint16 // approximate number of overflow buckets; see incrnoverflow for details
	   hash0     uint32 // hash seed
	   buckets    unsafe.Pointer // array of 2^B Buckets. may be nil if count==0.
	   oldbuckets unsafe.Pointer // previous bucket array of half the size, non-nil only when growing
	   nevacuate  uintptr        // progress counter for evacuation (buckets less than this have been evacuated)
	   extra *mapextra // optional fields
	}

比如當(dāng) B 為 5 的時(shí)候,通過獲取哈希值的低 5 位就能判斷出當(dāng)前鍵-元素對(duì)應(yīng)該存放在哪一個(gè)桶里面。例如我們通過哈希函數(shù),獲取到了一個(gè)鍵-元素對(duì)中鍵值的哈希值為

1001011100001111011011001000111100101010001001011001010101011011

其中,低 5 位代表其所屬的桶的位置,11011 換算為十進(jìn)制為 26 ,即該鍵-元素對(duì)存在第 26 個(gè)桶內(nèi)。哈希桶內(nèi)存儲(chǔ)的是“鍵的哈希值-內(nèi)部結(jié)構(gòu)”對(duì)的集合,即是按照 鍵1 鍵2 … 鍵8 元素1 元素2 … 元素8 溢出指針 的方式存儲(chǔ),是一塊連續(xù)的內(nèi)存,且鍵和元素時(shí)捆綁存儲(chǔ)的。我們找到哈希桶之后,再對(duì)比鍵值,就可以定位我們所以需要的鍵的位置,又因?yàn)殒I - 元素對(duì)是捆綁存儲(chǔ)的,所以找到了鍵就等于是找到對(duì)應(yīng)的元素值。

存儲(chǔ)時(shí)也是同樣的道理,但是要注意的是,每一個(gè)存儲(chǔ)桶最多只能存儲(chǔ) 8 個(gè)鍵-元素對(duì),當(dāng)超出 8 個(gè)的時(shí)候,就會(huì)生成一個(gè)溢出桶,并且當(dāng)前哈希桶的溢出指針(上述連續(xù)內(nèi)存的最后一塊)會(huì)指向新生成的溢出桶。

限制

其實(shí)從上面就可以看出,字典類型其實(shí)是一個(gè)哈希表的一個(gè)特定實(shí)現(xiàn),其中鍵和元素的最大區(qū)別在于鍵必須是可以哈希的,而元素卻可以是任意類型的,因此字典中的鍵類型是受限的。

字典聲明

// 聲明字典 是個(gè) nil 未初始化,直接存值會(huì)報(bào)錯(cuò)
var s0 map[string] int
// 聲明字典并初始化
s1 := map[string]int{}    
// 使用 make 聲明
s2 := make(map[string] int)
fmt.Println(s0, s1, s2, s3)

-------結(jié)果-------------------------
map[] map[] map[]

要注意:聲明字典的時(shí)候 key 的類型不能是函數(shù)、字典、切片。因?yàn)楦鶕?jù)上面查找字典鍵-元素對(duì)的過程可以知道,最后是要通過比較桶內(nèi)鍵和要查詢的鍵是不是一樣來確定鍵-元素對(duì)的位置的,但是這三種類型不支持判等操作,所以鍵的類型不支持這三種,編譯器會(huì)直接報(bào)錯(cuò)。

但是有一個(gè)比較特殊的類型:接口 interface{},interface{} 是支持判等操作的,所以編譯器不會(huì)報(bào)錯(cuò)。但是又因?yàn)?interface{} 這個(gè)空接口相當(dāng)于是個(gè)萬能類型,可以接受任何類型的值,所以會(huì)出現(xiàn)以下情況的代碼:

var s4 = map[interface{}]int{
	"1":      1,
	[]int{2}: 2,
	3:        3,
}
fmt.Println(s4)

------結(jié)果--------------
panic: runtime error: hash of unhashable type []int

當(dāng)我們運(yùn)行時(shí),就會(huì)出現(xiàn) panic 恐慌。程序運(yùn)行出現(xiàn)這樣的報(bào)錯(cuò)我們還能及時(shí)調(diào)整,但在程序運(yùn)行時(shí),我們添加了這樣的鍵值對(duì)進(jìn)去導(dǎo)致系統(tǒng)異常,再修改就為時(shí)已晚了,所以我們最好不要使用 interface{} 作為鍵的類型,而且我們要優(yōu)先考慮計(jì)算哈希值比較快的類型作為字典的鍵類型 。

字典賦值

//初始化
s0 := map[string]int{}
fmt.Println(s0)
//添加key-value
s0["one"] = 1
s0["two"] = 2
fmt.Println(s0)
//修改指定key的值
s0["one"] = 11
s0["two"] = 22
fmt.Println(s0)
//刪除指定key的元素
delete(s0, "one")
fmt.Println(s0)
//獲取key-value對(duì)個(gè)數(shù)
fmt.Println(len(s0))

------結(jié)果-------------------
map[]
map[one:1 two:2]
map[one:11 two:22]
map[two:22]
1

特殊類型修改值

如果值的類型是數(shù)組或者結(jié)構(gòu)體,那么不能直接修改 value 成員

s0 := map[string]struct {
	x int
}{}
s0["one"] = struct{ x int }{1}
s0["two"] = struct{ x int }{2}
s0["one"].x = 1 //這里編譯器會(huì)直接報(bào)錯(cuò)

方法一:先獲取全部value,修改之后重新賦值

s0 := map[string]struct {
	x int
}{}
s0["one"] = struct{ x int }{1}
s0["two"] = struct{ x int }{2}
s0["one"].x = 1 //這里編譯器會(huì)直接報(bào)錯(cuò)
// 正確做法一
s1 := s0["one"]
s1.x = 111
s0["one"] = s1 
fmt.Println(s0)

-----結(jié)果------------------
map[one:{111} two:{2}]

方法二:使用指針類型

* 開頭表示是指針類型

& 是取址符號(hào),即獲取對(duì)應(yīng)程序?qū)嶓w對(duì)象的地址

// 正確做法二 
// value 的類型是指針類型,指針指向結(jié)構(gòu)體
s0 := map[string]*struct {
	x int
}{}
//創(chuàng)建一個(gè)結(jié)構(gòu)體并把指針添加到字典中
s0["one"] = &struct{ x int }{1}
fmt.Println(*s0["one"])
s0["one"].x = 111
fmt.Println(*s0["one"])

-----結(jié)果------------------
{1}
{111}

字典遍歷

s0 := map[string]int{}
s0["one"] = 1
s0["two"] = 2
//接收 key 和 value
for k, vla := range s0 {
	fmt.Printf("%s:%d\n", k, vla)
}
fmt.Println("-----分割線---------------")
//只接收key
for k := range s0 {
	fmt.Printf("%s:%d\n", k, s0[k])
}

-----結(jié)果----------------
one:1
two:2
-----分割線---------------
one:1
two:2

總結(jié)字典特性

  • 字典的鍵類型是有限制的,必須支持哈希和判等
  • 字典是無序的,每次遍歷的順序都可能不一樣
  • 如果值類型是結(jié)構(gòu)體或者數(shù)組,那么不能直接對(duì)值的成員進(jìn)行操作
  • 不能對(duì) nil 字典進(jìn)行賦值操作,但是可以讀,讀出來是一個(gè)空字典 map[]
  • 字典是線程不安全的,多個(gè)線程對(duì)同一個(gè)字典進(jìn)行操作會(huì)導(dǎo)致報(bào)錯(cuò)
  • 可以在迭代過程中刪除或者添加鍵-元素對(duì)

到此這篇關(guān)于Go字典使用詳解的文章就介紹到這了,更多相關(guān)Go字典內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • 詳解Golang time包中的time.Duration類型

    詳解Golang time包中的time.Duration類型

    在日常開發(fā)過程中,會(huì)頻繁遇到對(duì)時(shí)間進(jìn)行操作的場(chǎng)景,使用 Golang 中的 time 包可以很方便地實(shí)現(xiàn)對(duì)時(shí)間的相關(guān)操作,本文講解一下 time 包中的 time.Duration 類型,需要的朋友可以參考下
    2023-07-07
  • 在Go中使用jwt的教程詳解

    在Go中使用jwt的教程詳解

    JWT (JSON Web Tokens) 是一種基于 JSON 格式的輕量級(jí)身份驗(yàn)證和授權(quán)方案,用于在各方之間以JSON方式安全地傳輸信息,本文給大家詳細(xì)介紹了在Go中使用jwt的教程,文中通過代碼示例講解的非常詳細(xì),需要的朋友可以參考下
    2024-06-06
  • 詳解如何使用Golang實(shí)現(xiàn)Cron定時(shí)任務(wù)

    詳解如何使用Golang實(shí)現(xiàn)Cron定時(shí)任務(wù)

    定時(shí)任務(wù)是許多應(yīng)用程序中常見的一種需求,它們可以用于執(zhí)行定期的清理任務(wù),發(fā)送通知,生成報(bào)告等,在這篇博客中,我們將介紹如何在Go語言中使用robfig/cron包來實(shí)現(xiàn)Cron定時(shí)任務(wù),需要的朋友可以參考下
    2024-04-04
  • Golang中Model的具體使用

    Golang中Model的具體使用

    本文主要介紹了Golang中Model的具體使用,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2023-04-04
  • go語言操作redis連接池的方法

    go語言操作redis連接池的方法

    這篇文章主要介紹了go語言操作redis連接池的方法,涉及Go語言操作radis的技巧,需要的朋友可以參考下
    2015-03-03
  • 一文讓你理解go語言的Context

    一文讓你理解go語言的Context

    在Go語言中,Context(上下文)是一個(gè)類型,用于在程序中傳遞請(qǐng)求范圍的值、截止時(shí)間、取消信號(hào)和其他與請(qǐng)求相關(guān)的上下文信息,它在多個(gè)goroutine之間傳遞這些值,使得并發(fā)編程更加可靠和簡(jiǎn)單,本文詳細(xì)介紹go語言的Context,需要的朋友可以參考下
    2023-05-05
  • Go習(xí)慣用法(多值賦值短變量聲明賦值簡(jiǎn)寫模式)基礎(chǔ)實(shí)例

    Go習(xí)慣用法(多值賦值短變量聲明賦值簡(jiǎn)寫模式)基礎(chǔ)實(shí)例

    本文為大家介紹了Go習(xí)慣用法(多值賦值,短變量聲明和賦值,簡(jiǎn)寫模式、多值返回函數(shù)、comma,ok 表達(dá)式、傳值規(guī)則)的基礎(chǔ)實(shí)例,幫大家鞏固扎實(shí)Go語言基礎(chǔ)
    2024-01-01
  • go redis實(shí)現(xiàn)滑動(dòng)窗口限流的方式(redis版)

    go redis實(shí)現(xiàn)滑動(dòng)窗口限流的方式(redis版)

    這篇文章主要介紹了go redis實(shí)現(xiàn)滑動(dòng)窗口限流的方式(redis版),本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2020-12-12
  • GO workPool的線程池實(shí)現(xiàn)

    GO workPool的線程池實(shí)現(xiàn)

    本文主要介紹了GO workPool的線程池實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2023-03-03
  • Centos下搭建golang環(huán)境及vim高亮Go關(guān)鍵字設(shè)置的方法

    Centos下搭建golang環(huán)境及vim高亮Go關(guān)鍵字設(shè)置的方法

    這篇文章先給大家詳細(xì)介紹了在Centos下搭建golang環(huán)境的步驟,大家按照下面的方法就可以自己搭建golang環(huán)境,搭建完成后又給大家介紹了vim高亮Go關(guān)鍵字設(shè)置的方法,文中通過示例代碼介紹的很詳細(xì),有需要的朋友們可以參考借鑒,下面來一起看看吧。
    2016-11-11

最新評(píng)論