主頁 > 後端開發 > 包含參考型別欄位的自定義結構體,能作為map的key嗎

包含參考型別欄位的自定義結構體,能作為map的key嗎

2023-06-05 07:51:00 後端開發

1. 引言

在 Go 語言中,map是一種內置的資料型別,它提供了一種高效的方式來存盤和檢索資料,map是一種無序的鍵值對集合,其中每個鍵與一個值相關聯,使用 map 資料結構可以快速地根據鍵找到對應的值,而無需遍歷整個集合,

在 Go 語言中,map 是一種內置的資料型別,可以通過以下方式宣告和初始化:

m := make(map[keyType]valueType)

在使用map時,我們通常會使用基本資料型別作為鍵,然而,當我們需要將自定義的結構體作為鍵時,就需要考慮結構體中是否包含參考型別的欄位,參考型別是指存盤了資料的地址的型別,如指標、切片、字典和通道等,在Go中,參考型別具有動態的特性,可能會被修改或指向新的資料,這就引發了一個問題:能否將包含參考型別的自定義結構體作為map的鍵呢?

2. map的基本模型

了解能否將包含參考型別的自定義結構體作為map的鍵這個問題,我們需要先了解下map的基本模型,在Go語言中,map是使用哈希表、實作的,哈希表是一種以鍵-值對形式存盤資料的資料結構,它通過使用哈希函式將鍵映射到哈希值,

哈希函式是用于將鍵映射到哈希值的演算法,它接受鍵作為輸入并生成一個固定長度的哈希值,Go語言的 map 使用了內部的哈希函式來計算鍵的哈希值,

而不同的key通過哈希函式生成的哈希值可能是相同的,此時便發生了哈希沖突,哈希沖突指的是不同的鍵經過哈希函式計算后得到相同的哈希值,由于哈希函式的輸出空間遠遠小于鍵的輸入空間,哈希沖突是不可避免的,此時無法判斷該key是當前哈希表中原本便已經存在的元素還是由于哈希沖突導致不同的鍵映射到同一個bucket, 此時便需要判斷這兩個key是否相等,

因此,在map中,作為map中的key,需要保證其支持對比操作的,能夠比較兩個key是否相等,

3. map 鍵的要求

從上面map基本的模型介紹中,我們了解到,map中的Key需要支持哈希函式的計算,同時鍵的型別必須支持對比操作,

map中,計算key的哈希值,是由默認哈希函式實作的,對于map中的key并沒有額外的要求,

map中,判斷兩個鍵是否相等是通過呼叫鍵型別的相等運算子(==!=)來完成的,因此key必須確保該型別支持 == 操作,這個要求是由 map 的實作機制決定的,map 內部使用鍵的相等性來確定鍵的存盤位置和檢索值,如果鍵的型別不可比較,就無法進行相等性比較,從而導致無法準確地定位鍵和檢索值,

在 Go 中,基本資料型別(如整數、浮點數、字串)和一些內置型別都是可比較的,因此它們可以直接用作 map 的鍵,然而,自定義的結構體作為鍵時,需要確保結構體的所有欄位都是可比較的型別,如果結構體包含參考型別的欄位,那么該結構體就不能直接用作 map 的鍵,因為參考型別不具備簡單的相等性比較,

因此,假如map中的鍵為自定義型別,同時包含參考欄位,此時將無法作為map的鍵,會直接編譯失敗,代碼示例如下:

type Person struct {
   Name    string
   Age     int
   address []Address
}
func main() {
    // 這里會直接編譯不通過
    m := make(map[Person]int)
}

其次還有一個例外,那便是自定義結構體中包含指標型別的欄位,此時其是支持==操作的,但是其是使用指標地址來進行hash計算以及相等性比較的,有可能我們理解是同一個key,事實上從map來看并不是,此時非常容易導致錯誤,示例如下:

type Person struct {
   Name    string
   Age     int
   address *Address
}
func main(){
    m := make(map[Person]int)
    p1 := Person{Name: "Alice", Age: 30, address: &Address{city: "beijing"}}
    p2 := Person{Name: "Alice", Age: 30, address: &Address{city: "beijing"}}
    m[p1] = 1
    m[p2] = 2
    // 輸出1
    fmt.Println(m[p1])
    // 輸出2
    fmt.Println(m[p2])
}

這里我們定義了一個Person結構體,包含一個指標型別的欄位address,創建了兩個物件p1p2,在我們的理解中,其是同一個物件,事實上在map中為兩個兩個互不相關的物件,主要原因都是使用地址來進行hash計算以及相等性比較的,

綜上所述,如果自定義結構體中包含參考型別的欄位(指標為特殊的參考型別),此時將不能作為map型別的key

4. 為什么不抽取hashCode和equals方法介面,由用戶自行實作呢?

當前gomap中哈希值的計算,其提供了默認的哈希函式,不需要由用戶去實作;其次key的相等性比較,是通過== 運算子來實作的,也不由用戶自定義比較函式,那我們就有一個疑問了,為什么不抽取hashCode和equals方法介面,由用戶來實作呢?

4.1 簡單性和性能角度

相等性比較在 Go 語言中使用 == 運算子來實作,而哈希函式是由運行時庫提供的默認實作,這種設計選擇我理解可能基于以下幾個原因:

  1. 簡單性:對于默認哈希函式函式來說,其內置在語言中的,無需用戶額外的實作和配置,這簡化了 map 的使用,對于相等性比較操作,== 運算子進行比較是一種直觀且簡單的方式,在語法上,== 運算子用于比較兩個值是否相等,這種語法的簡潔性使得代碼更易讀和理解,
  2. 性能:默認的哈希函式是經過優化和測驗的,能夠在大多數情況下提供良好的性能,其次使用==來實作相等性比較,由于 == 運算子是語言層面的原生操作,編譯器可以對其進行優化,從而提高代碼的執行效率,

4.2 key不可變的限制

map鍵的不可變性也是一個考慮因素,基于==來判斷物件是否相等,間接保證了鍵的不可變性,目前,==已經支持了大部分型別的比較,只有自定義結構體中的參考型別欄位無法直接使用==進行比較,如果鍵中不存在參考型別欄位,這意味著放入Map鍵的值在運行時不能發生變化,從而保證了鍵在運行時的不可變性,

如果key沒有不可變的限制,那么之前存盤在 map 中的鍵值對可能會出現問題,因為在放置元素時,map 會根據鍵的當前值計算哈希值,并使用哈希值來查找對應的存盤位置,如果放在map中的鍵的值發生了變化,此時計算出來的hash值可能也發生變化,這意味資料放在了錯誤的位置,后續即使使用跟map中的鍵的同一個值去查找資料,也可能查找不到資料,

下面展示一個簡單的代碼,來說明可變型別作為key會導致的問題:

type Person struct {
    Name       string
    Age        int
    SliceField []string
}

func main() {
    person := Person{Name: "Alice", Age: 25, SliceField: []string{"A", "B"}}
    // 假設Person可以作為鍵,事實上是不支持的
    personMap := make(map[Person]string)
    personMap[person] = "Value 1"

    // 修改person中SliceField的值
    person.SliceField[0] = "X"

    // 嘗試通過相同的person查找值
    fmt.Println(personMap[person]) // 輸出空字串,找不到對應的值
}

如果抽取equals方法介面,由用戶自行實作,此時key的不可變性就需要用戶實作,其次go語言也需要增加一些檢測機制,這首先增加了用戶使用的負擔,這并不符合go語言設計的哲學,

4.3 總結

綜上所述,基于簡單性、性能和語意一致性的考慮以及鍵的不可變性,Go語言選擇使用==運算子進行鍵的比較,而將哈希函式作為運行時庫的默認實作,更加符合go語言設計的哲學,

5. 總結

在 Go 語言中,map 是一種無序的鍵值對集合,它提供了高效的資料存盤和檢索機制,在使用 map 時,通常使用基本資料型別作為鍵,然而,當我們想要使用自定義結構體作為鍵時,需要考慮結構體中是否包含參考型別的欄位,

自定義結構體作為map的鍵需要滿足一些要求,首先,鍵的型別必須是可比較的,也就是支持通過== 運算子進行相等性比較,在Go中,基本資料型別和一些內置型別都滿足這個要求,但是,如果結構體中包含參考型別的欄位,那么該結構體就不能直接作為map的鍵,因為參考型別不具備簡單的相等性比較,

因此總的來說,包含參考型別欄位的自定義結構體,是不能作為mapkey的,

轉載請註明出處,本文鏈接:https://www.uj5u.com/houduan/554258.html

標籤:其他

上一篇:p3 FileInputStream 和 FileOutputStream

下一篇:返回列表

標籤雲
其他(160311) Python(38201) JavaScript(25475) Java(18185) C(15236) 區塊鏈(8269) C#(7972) AI(7469) 爪哇(7425) MySQL(7231) html(6777) 基礎類(6313) sql(6102) 熊猫(6058) PHP(5873) 数组(5741) R(5409) Linux(5346) 反应(5209) 腳本語言(PerlPython)(5129) 非技術區(4971) Android(4582) 数据框(4311) css(4259) 节点.js(4032) C語言(3288) json(3245) 列表(3129) 扑(3119) C++語言(3117) 安卓(2998) 打字稿(2995) VBA(2789) Java相關(2746) 疑難問題(2699) 细绳(2522) 單片機工控(2479) iOS(2434) ASP.NET(2403) MongoDB(2323) 麻木的(2285) 正则表达式(2254) 字典(2211) 循环(2198) 迅速(2185) 擅长(2169) 镖(2155) .NET技术(1981) 功能(1967) HtmlCss(1952) Web開發(1951) C++(1928) python-3.x(1918) 弹簧靴(1913) xml(1889) PostgreSQL(1879) .NETCore(1863) 谷歌表格(1846) Unity3D(1843) for循环(1842)

熱門瀏覽
  • 【C++】Microsoft C++、C 和匯編程式檔案

    ......

    uj5u.com 2020-09-10 00:57:23 more
  • 例外宣告

    相比于斷言適用于排除邏輯上不可能存在的狀態,例外通常是用于邏輯上可能發生的錯誤。 例外宣告 Item 1:當函式不可能拋出例外或不能接受拋出例外時,使用noexcept 理由 如果不打算拋出例外的話,程式就會認為無法處理這種錯誤,并且應當盡早終止,如此可以有效地阻止例外的傳播與擴散。 示例 //不可 ......

    uj5u.com 2020-09-10 00:57:27 more
  • Codeforces 1400E Clear the Multiset(貪心 + 分治)

    鏈接:https://codeforces.com/problemset/problem/1400/E 來源:Codeforces 思路:給你一個陣列,現在你可以進行兩種操作,操作1:將一段沒有 0 的區間進行減一的操作,操作2:將 i 位置上的元素歸零。最終問:將這個陣列的全部元素歸零后操作的最少 ......

    uj5u.com 2020-09-10 00:57:30 more
  • UVA11610 【Reverse Prime】

    本人看到此題沒有翻譯,就附帶了一個自己的翻譯版本 思考 這一題,它的第一個要求是找出所有 $7$ 位反向質數及其質因數的個數。 我們應該需要質數篩篩選1~$10^{7}$的所有數,這里就不慢慢介紹了。但是,重讀題,我們突然發現反向質數都是 $7$ 位,而將它反過來后的數字卻是 $6$ 位數,這就說明 ......

    uj5u.com 2020-09-10 00:57:36 more
  • 統計區間素數數量

    1 #pragma GCC optimize(2) 2 #include <bits/stdc++.h> 3 using namespace std; 4 bool isprime[1000000010]; 5 vector<int> prime; 6 inline int getlist(int ......

    uj5u.com 2020-09-10 00:57:47 more
  • C/C++編程筆記:C++中的 const 變數詳解,教你正確認識const用法

    1、C中的const 1、區域const變數存放在堆疊區中,會分配記憶體(也就是說可以通過地址間接修改變數的值)。測驗代碼如下: 運行結果: 2、全域const變數存放在只讀資料段(不能通過地址修改,會發生寫入錯誤), 默認為外部聯編,可以給其他源檔案使用(需要用extern關鍵字修飾) 運行結果: ......

    uj5u.com 2020-09-10 00:58:04 more
  • 【C++犯錯記錄】VS2019 MFC添加資源不懂如何修改資源宏ID

    1. 首先在資源視圖中,添加資源 2. 點擊新添加的資源,復制自動生成的ID 3. 在解決方案資源管理器中找到Resource.h檔案,編輯,使用整個專案搜索和替換的方式快速替換 宏宣告 4. Ctrl+Shift+F 全域搜索,點擊查找全部,然后逐個替換 5. 為什么使用搜索替換而不使用屬性視窗直 ......

    uj5u.com 2020-09-10 00:59:11 more
  • 【C++犯錯記錄】VS2019 MFC不懂的批量添加資源

    1. 打開資源頭檔案Resource.h,在其中預先定義好宏 ID(不清楚其實ID值應該設定多少,可以先新建一個相同的資源項,再在這個資源的ID值的基礎上遞增即可) 2. 在資源視圖中選中專案資源,按F7編輯資源檔案,按 ID 型別 相對路徑的形式添加 資源。(別忘了先把檔案拷貝到專案中的res檔案 ......

    uj5u.com 2020-09-10 01:00:19 more
  • C/C++編程筆記:關于C++的參考型別,專供新手入門使用

    今天要講的是C++中我最喜歡的一個用法——參考,也叫別名。 參考就是給一個變數名取一個變數名,方便我們間接地使用這個變數。我們可以給一個變數創建N個參考,這N + 1個變數共享了同一塊記憶體區域。(參考型別的變數會占用記憶體空間,占用的記憶體空間的大小和指標型別的大小是相同的。雖然參考是一個物件的別名,但 ......

    uj5u.com 2020-09-10 01:00:22 more
  • 【C/C++編程筆記】從頭開始學習C ++:初學者完整指南

    眾所周知,C ++的學習曲線陡峭,但是花時間學習這種語言將為您的職業帶來奇跡,并使您與其他開發人員區分開。您會更輕松地學習新語言,形成真正的解決問題的技能,并在編程的基礎上打下堅實的基礎。 C ++將幫助您養成良好的編程習慣(即清晰一致的編碼風格,在撰寫代碼時注釋代碼,并限制類內部的可見性),并且由 ......

    uj5u.com 2020-09-10 01:00:41 more
最新发布
  • 包含參考型別欄位的自定義結構體,能作為map的key嗎

    # 1. 引言 在 Go 語言中,`map`是一種內置的資料型別,它提供了一種高效的方式來存盤和檢索資料。`map`是一種無序的鍵值對集合,其中每個鍵與一個值相關聯。使用 map 資料結構可以快速地根據鍵找到對應的值,而無需遍歷整個集合。 在 Go 語言中,`map` 是一種內置的資料型別,可以通過 ......

    uj5u.com 2023-06-05 07:51:00 more
  • p3 FileInputStream 和 FileOutputStream

    # FileInputStream 和 FileOutputStream ![](https://img2023.cnblogs.com/blog/3008601/202306/3008601-20230604102221520-1382311786.png) - InputStream:位元組輸入流 ......

    uj5u.com 2023-06-05 07:50:48 more
  • p2 IO流原理及流的分類

    # IO流原理及流的分類 ### 一、Java IO流原理 1. I/O是Input/Output的縮寫,I/O技術是非常實用的技術,用于處理資料傳輸。如讀/寫檔案,網路通訊等。 2. Java程式中,對于資料的輸入/輸出操作以”流(stream)“的方式進行。 3. java.io包下提供了各種” ......

    uj5u.com 2023-06-05 07:45:25 more
  • FHQ-Treap

    [模板傳送](https://www.luogu.com.cn/problem/P3369) FHQ-Treap顧名思義就是范浩強大佬設計的一種二叉平衡樹 下面我們來講一下它的原理和代碼 # 結構體 對于一個節點,我們需要記錄的是 * 對應的值 * 子樹節點數 * 左右孩子編號 * 對應的隨機值 ` ......

    uj5u.com 2023-06-05 07:40:15 more
  • p1 檔案的基本使用

    # 檔案的基本使用 ### 一、檔案 - **什么是檔案** 檔案是保存資料的地方,比如word檔案,txt檔案,excel檔案……都是檔案。即可以保存一張圖片,也可以保持視頻,聲音…… - **檔案流** 檔案在程式中是以流的形式來操作的 ![檔案流](https://img2023.cnblog ......

    uj5u.com 2023-06-05 07:40:06 more
  • Groovy 基于Groovy實作DES加解密

    groovy 3.0.7 ### DES加密簡介 加密分為對稱加密和非對稱加密。非對稱加密,加解密使用不同的密鑰,如RSA;對稱加密,加解密使用相同的密鑰,如DES(Data Encryption Standard,即資料加密標準)。相對而言,非對稱加密安全性更高,但是計算程序復雜耗時,一般只應用于 ......

    uj5u.com 2023-06-05 07:39:54 more
  • 【python基礎】復雜資料型別-串列型別(串列切片)

    # 1.串列切片 前面學習的是如何處理串列的所有資料元素。python還可以處理串列的部分元素,python稱之為切片。 ## 1.1創建切片 創建切片,可指定要使用的第一個資料元素的索引和最后一個資料元素的索引。與range函式一樣,python在到達指定的第二個索引前面的資料元素后停止。比如要輸 ......

    uj5u.com 2023-06-05 07:34:02 more
  • 基于ESP32的TCP/IP傳輸實作

    #TCP/IP協議原理 TCP/IP協議是Internet互聯網最基本的協議,TCP/IP協議的應用層的主要協議有HTTP、Telnet、FTP、SMTP等,是用來讀取來自傳輸層的資料或者將資料傳輸寫入傳輸層;傳輸層的主要協議有UDP、TCP,實作端對端的資料傳輸;網路層的主要協議有ICMP、IP、 ......

    uj5u.com 2023-06-04 07:46:19 more
  • 【python基礎】復雜資料型別-串列型別(排序/長度/遍歷)

    # 1.串列資料元素排序 在創建的串列中,資料元素的排列順序常常是無法預測的。這雖然在大多數情況下都是不可避免的,但經常需要以特定的順序呈現資訊。有時候希望保留串列資料元素最初的排列順序,而有時候又需要調整排列順序。python提供了很多串列資料元素排序的方式,可根據情況選用。 ## 1.永久性排序 ......

    uj5u.com 2023-06-04 07:45:59 more
  • 【VS Code 與 Qt6】QCheckBox的圖示為什么不會切換?

    本篇專門扯一下有關 QCheckBox 組件的一個問題。老周不水字數,直接上程式,你看了就明白。 #include <QApplication> #include <QWidget> #include <QPushButton> #include <QCheckBox> #include <QVBo ......

    uj5u.com 2023-06-04 07:45:38 more