欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

golang 如何獲取map所有key的方式

 更新時(shí)間:2021年04月25日 09:22:33   作者:YZF_Kevin  
這篇文章主要介紹了golang 獲取map所有key的方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧

最佳方式:根據(jù)map的長(zhǎng)度,新建一個(gè)數(shù)組,遍歷map逐個(gè)壓入

方法1(效率很高):

func getKeys1(m map[int]int) []int {
 // 數(shù)組默認(rèn)長(zhǎng)度為map長(zhǎng)度,后面append時(shí),不需要重新申請(qǐng)內(nèi)存和拷貝,效率很高
 j := 0
 keys := make([]int, len(m))
 for k := range m {
  keys[j] = k
  j++
 }
 return keys
}

方法2(效率很高):

func getKeys2(m map[int]int) []int {
 // 數(shù)組默認(rèn)長(zhǎng)度為map長(zhǎng)度,后面append時(shí),不需要重新申請(qǐng)內(nèi)存和拷貝,效率很高
 keys := make([]int, 0, len(m))
 for k := range m {
  keys = append(keys, k)
 }
 return keys
}

其他方式:

方法3(效率較低):

func getKeys3(m map[int]int) []int {
 // 注意:由于數(shù)組默認(rèn)長(zhǎng)度為0,后面append時(shí),需要重新申請(qǐng)內(nèi)存和拷貝,所以效率較低
 keys := []int{}
 for k := range m {
  keys = append(keys, k)
 }
 return keys
}

方法4(效率極低):

func getKeys4(m map[int]int) int {
 // 注意:雖然此寫法簡(jiǎn)潔,但MapKeys函數(shù)內(nèi)部操作復(fù)雜,效率極低
 keys := reflect.ValueOf(m).MapKeys()
 return len(keys)
}

實(shí)驗(yàn)結(jié)果如圖(可以看到方法1和方法2效率最高,內(nèi)存操作也最少):

完整代碼如下:

package test
 
import (
 "reflect"
 "testing"
)
 
// 初始化map
func initMap() map[int]int {
 m := map[int]int{}
 for i := 0; i < 10000; i++ {
  m[i] = i
 }
 return m
}
 
func getKeys1(m map[int]int) []int {
 // 數(shù)組默認(rèn)長(zhǎng)度為map長(zhǎng)度,后面append時(shí),不需要重新申請(qǐng)內(nèi)存和拷貝,效率較高
 j := 0
 keys := make([]int, len(m))
 for k := range m {
  keys[j] = k
  j++
 }
 return keys
}
 
func getKeys2(m map[int]int) []int {
 // 數(shù)組默認(rèn)長(zhǎng)度為map長(zhǎng)度,后面append時(shí),不需要重新申請(qǐng)內(nèi)存和拷貝,效率較高
 keys := make([]int, 0, len(m))
 for k := range m {
  keys = append(keys, k)
 }
 return keys
}
 
// 初始化默認(rèn)
func getKeys3(m map[int]int) []int {
 // 注意:由于數(shù)組默認(rèn)長(zhǎng)度為0,后面append時(shí),需要重新申請(qǐng)內(nèi)存和拷貝,所以效率較低
 keys := []int{}
 for k := range m {
  keys = append(keys, k)
 }
 return keys
}
 
// 使用反射
func getKeys4(m map[int]int) int {
 // 注意:雖然此寫法簡(jiǎn)潔,但MapKeys函數(shù)內(nèi)部操作復(fù)雜,效率極低
 keys := reflect.ValueOf(m).MapKeys()
 return len(keys)
}
 
func BenchmarkMapkeys1(b *testing.B) {
 // 初始化map
 m := initMap()
 
 b.ResetTimer()
 for i := 0; i < b.N; i++ {
  getKeys1(m)
 }
}
func BenchmarkMapkeys2(b *testing.B) {
 // 初始化map
 m := initMap()
 
 b.ResetTimer()
 for i := 0; i < b.N; i++ {
  getKeys2(m)
 }
}
 
func BenchmarkMapkeys3(b *testing.B) {
 // 初始化map
 m := initMap()
 
 b.ResetTimer()
 for i := 0; i < b.N; i++ {
  getKeys3(m)
 }
}
 
func BenchmarkMapkeys4(b *testing.B) {
 // 初始化map
 m := initMap()
 
 b.ResetTimer()
 for i := 0; i < b.N; i++ {
  getKeys4(m)
 }
}

補(bǔ)充:Golang踩坑——判斷map中是否有key

最近在實(shí)習(xí),下班回去十點(diǎn)多了,再加上比較懶,快有兩個(gè)月沒寫東西了。

今天在開發(fā)一個(gè)模塊的時(shí)候是接著上一個(gè)人的寫的,好不容易各種配置寫好了開始跑,發(fā)現(xiàn)他踩了一個(gè)很容易踩的坑。

把bug抽出來單獨(dú)寫了個(gè)文件

package main
 
import (
	"log"
)
 
type agent struct {
	id  int
	str string
}
 
var m map[int]*agent 
func main() {
	m = make(map[int]*agent)
	a := &agent{
		id:  1,
		str: "hello",
	}
	log.Println(a)
	var ok bool
	if a, ok = m[1]; ok {
		log.Println("ok")
	}
	m[1] = a
	log.Println(a)
	log.Println(a.str)
}

運(yùn)行一下

可以看到報(bào)了無效指針的問題(invalid memory address or nil pointer dereference)。

從上面的兩個(gè)log可以看到,最開始指針是有值的,第二個(gè)卻沒了,原因就出現(xiàn)在那個(gè)if判斷那里。

if判斷是想通過查看map知道agent是否已經(jīng)存在,這個(gè)地方的錯(cuò)誤是a,ok = map; 我們想通過ok判斷是否存在,但是在這里還有個(gè)a,這會(huì)導(dǎo)致有一個(gè)對(duì)a賦值的操作,也就是說如果map里有這個(gè)值,那沒事,如果沒有的話a就被賦值成了nil了。

在下面打印的時(shí)候也就成了nil,取它的屬性的時(shí)候會(huì)告訴我們找了個(gè)空指針的屬性,這肯定找不到。

我們把a(bǔ)換成_,再來看結(jié)果:

ok了。

以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教。

相關(guān)文章

  • Golang Gorm 更新字段save、update、updates

    Golang Gorm 更新字段save、update、updates

    在gorm中,批量更新操作可以通過使用Update方法來實(shí)現(xiàn),本文主要介紹了Golang Gorm 更新字段save、update、updates,具有一定的參考價(jià)值,感興趣的可以了解一下
    2023-12-12
  • 使用docker構(gòu)建golang線上部署環(huán)境的步驟詳解

    使用docker構(gòu)建golang線上部署環(huán)境的步驟詳解

    這篇文章主要介紹了使用docker構(gòu)建golang線上部署環(huán)境的步驟,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧。
    2017-11-11
  • Go 1.18新特性之泛型的全面講解

    Go 1.18新特性之泛型的全面講解

    本文力求能讓未接觸過泛型編程的人也能較好理解Go的泛型,所以行文可能略顯啰嗦。但是請(qǐng)相信我,看完這篇文章你能獲得對(duì)Go泛型非常全面的了解
    2023-03-03
  • Go語言sync.Cond使用方法詳解

    Go語言sync.Cond使用方法詳解

    Go語言標(biāo)準(zhǔn)庫(kù)中還包含條件變量 sync.Cond,它可以讓一組 Goroutine 都在滿足特定條件時(shí)被喚醒,每一個(gè)sync.Cond結(jié)構(gòu)體在初始化時(shí)都需要傳入一個(gè)互斥鎖,接下來我們將通過文中例子了解它的使用方法,感興趣的同學(xué)跟著小編一起來看看吧
    2023-07-07
  • 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
  • Golang設(shè)計(jì)模式之外觀模式講解和代碼示例

    Golang設(shè)計(jì)模式之外觀模式講解和代碼示例

    外觀是一種結(jié)構(gòu)型設(shè)計(jì)模式, 能為復(fù)雜系統(tǒng)、 程序庫(kù)或框架提供一個(gè)簡(jiǎn)單 (但有限) 的接口,這篇文章就給大家詳細(xì)介紹一下Golang的外觀模式,文中有詳細(xì)的代碼示例,具有一定的參考價(jià)值,需要的朋友可以參考下
    2023-06-06
  • vim配置go語言語法高亮問題的解決方法

    vim配置go語言語法高亮問題的解決方法

    vim配置go語言語法高亮的問題已經(jīng)遇到過好幾次了,每次都是找不到答案,今天小編給大家?guī)砹藇im配置go語言語法高亮問題的解決方法,感興趣的朋友一起看看吧
    2018-01-01
  • 詳解golang中接口使用的最佳時(shí)機(jī)

    詳解golang中接口使用的最佳時(shí)機(jī)

    接口在系統(tǒng)設(shè)計(jì)中,以及代碼重構(gòu)優(yōu)化中,是一個(gè)不可或缺的工具,能夠幫助我們寫出可擴(kuò)展,可維護(hù)性更強(qiáng)的程序,本文主要為大家介紹一下golang中接口使用的最佳時(shí)機(jī),有興趣的可以了解下
    2023-09-09
  • golang中json和struct的使用說明

    golang中json和struct的使用說明

    這篇文章主要介紹了golang中json和struct的使用說明,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧
    2021-05-05
  • Go中的fuzz模糊測(cè)試使用實(shí)戰(zhàn)詳解

    Go中的fuzz模糊測(cè)試使用實(shí)戰(zhàn)詳解

    這篇文章主要為大家介紹了Go中的fuzz模糊測(cè)試使用實(shí)戰(zhàn)詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-12-12

最新評(píng)論