包含引用類型欄位的自定義結構體,能作為map的key嗎

来源:https://www.cnblogs.com/chenjiazhan/archive/2023/06/04/17455649.html
-Advertisement-
Play Games

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


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的。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • # SpringCloud Gateway-服務網關 ## 1.Gateway介紹 ### 1.1引出問題 **沒有使用網關服務時:** **使用網關服務後:** ### 1.2Gateway網路拓撲圖 ![Gateway網路拓撲圖](https://liyuelian.oss-cn-shenzhe ...
  • # 1.初識元組 列表非常適合用於存儲在程式運行期間可能變化的數據集。列表是可以修改的。 然而,有時候需要創建一系列不可修改的元素,元組可以滿足這種需求 python將不能修改的值稱為不可變的,而不可變的列表被稱為元組。 元組看起來猶如列表,但使用圓括弧而不是方括弧來標識。 其語法格式:元組變數名 ...
  • 在`pandas`中,索引(`index`)是用於訪問數據的關鍵。 它為數據提供了基於標簽的訪問能力,類似於字典,可以根據標簽查找和訪問數據。 而`pandas`的軸(`axis`)是指數據表中的一個維度,可以理解為表格中的行和列。 通過指定軸,我們可以對數據進行切片、篩選、聚合等操作。 下麵簡要介 ...
  • # FileReader 和 FileWriter ### 一、 FileReader 和 File Writer 介紹 FileReader 和 FileWriter 是字元流,即按照字元來操作 io ### 二、 FileReader 相關方法 ![](https://img2023.cnblo ...
  • ## 1. 環境配置 - Springboot 2.7.8 - h2 2.1.214 ## 2. POM文件 - 引入springboot parent pom 點擊查看代碼 ``` org.springframework.boot spring-boot-starter-parent 2.7.8 ...
  • 某日小二參加XXX科技公司的C++工程師開發崗位5面: > 面試官:struct和class有什麼區別? > > 小二:在C++中,struct和class的唯一區別是預設的訪問控制。struct預設的成員是public的,而class的預設成員是private的。 > > 面試官:struct、c ...
  • # 前言 最近在開發文件存儲服務,需要符合s3的協議標準,可以直接接入aws-sdk,本文針對sdk發出請求的鑒權信息進行重新組合再簽名驗證有效性,sdk版本如下 ```xml software.amazon.awssdk s3 2.20.45 ``` # 演算法解析 首先對V4版本簽名演算法的數據結構 ...
  • # Rust Web 全棧開發之編寫 WebAssembly 應用 MDN Web Docs: 官網: ## 項目結構 和 功能 **Web App 教師註冊 WebService WebAssembly App 課程管理** ## 什麼是 WebAssembly - WebAssembly 是一種 ...
一周排行
    -Advertisement-
    Play Games
  • 前言 在我們開發過程中基本上不可或缺的用到一些敏感機密數據,比如SQL伺服器的連接串或者是OAuth2的Secret等,這些敏感數據在代碼中是不太安全的,我們不應該在源代碼中存儲密碼和其他的敏感數據,一種推薦的方式是通過Asp.Net Core的機密管理器。 機密管理器 在 ASP.NET Core ...
  • 新改進提供的Taurus Rpc 功能,可以簡化微服務間的調用,同時可以不用再手動輸出模塊名稱,或調用路徑,包括負載均衡,這一切,由框架實現並提供了。新的Taurus Rpc 功能,將使得服務間的調用,更加輕鬆、簡約、高效。 ...
  • 順序棧的介面程式 目錄順序棧的介面程式頭文件創建順序棧入棧出棧利用棧將10進位轉16進位數驗證 頭文件 #include <stdio.h> #include <stdbool.h> #include <stdlib.h> 創建順序棧 // 指的是順序棧中的元素的數據類型,用戶可以根據需要進行修改 ...
  • 前言 整理這個官方翻譯的系列,原因是網上大部分的 tomcat 版本比較舊,此版本為 v11 最新的版本。 開源項目 從零手寫實現 tomcat minicat 別稱【嗅虎】心有猛虎,輕嗅薔薇。 系列文章 web server apache tomcat11-01-官方文檔入門介紹 web serv ...
  • C總結與剖析:關鍵字篇 -- <<C語言深度解剖>> 目錄C總結與剖析:關鍵字篇 -- <<C語言深度解剖>>程式的本質:二進位文件變數1.變數:記憶體上的某個位置開闢的空間2.變數的初始化3.為什麼要有變數4.局部變數與全局變數5.變數的大小由類型決定6.任何一個變數,記憶體賦值都是從低地址開始往高地 ...
  • 如果讓你來做一個有狀態流式應用的故障恢復,你會如何來做呢? 單機和多機會遇到什麼不同的問題? Flink Checkpoint 是做什麼用的?原理是什麼? ...
  • C++ 多級繼承 多級繼承是一種面向對象編程(OOP)特性,允許一個類從多個基類繼承屬性和方法。它使代碼更易於組織和維護,並促進代碼重用。 多級繼承的語法 在 C++ 中,使用 : 符號來指定繼承關係。多級繼承的語法如下: class DerivedClass : public BaseClass1 ...
  • 前言 什麼是SpringCloud? Spring Cloud 是一系列框架的有序集合,它利用 Spring Boot 的開發便利性簡化了分散式系統的開發,比如服務註冊、服務發現、網關、路由、鏈路追蹤等。Spring Cloud 並不是重覆造輪子,而是將市面上開發得比較好的模塊集成進去,進行封裝,從 ...
  • class_template 類模板和函數模板的定義和使用類似,我們已經進行了介紹。有時,有兩個或多個類,其功能是相同的,僅僅是數據類型不同。類模板用於實現類所需數據的類型參數化 template<class NameType, class AgeType> class Person { publi ...
  • 目錄system v IPC簡介共用記憶體需要用到的函數介面shmget函數--獲取對象IDshmat函數--獲得映射空間shmctl函數--釋放資源共用記憶體實現思路註意 system v IPC簡介 消息隊列、共用記憶體和信號量統稱為system v IPC(進程間通信機制),V是羅馬數字5,是UNI ...