0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

struct,slice,map是否相等以及幾種對比方法的區(qū)別

馬哥Linux運維 ? 來源:稀土掘金 ? 2024-01-08 18:20 ? 次閱讀

一、前言

對比兩個struct或者map,slice是否相等是大家經(jīng)常會有的需求,想必大家也都接觸過很多對比的方式,比如==,reflect.DeepEqual(),cmp.Equal()等。

這么多種對比方式,適用場景和優(yōu)缺點都有哪些呢?為什么可以用==,有的卻不可以呢?除了這三個,還有其他的方式可以判斷相等嗎?問題多多,且一起研究研究。

二、== 的對比方式

1、golang的四大類型

golang中的數(shù)據(jù)類型可以分為以下 4 大類:


基本類型:整型(
int/uint/int8/uint8/int16/uint16/int32/uint32/int64/uint64/byte/rune等)、浮點數(shù)(
float32/float64)、復數(shù)類型(
complex64/complex128)、字符串(
string)。
復合類型(又叫聚合類型):數(shù)組和結構體類型。
引用類型:切片(slice)、map、channel、指針。
接口類型:如error :類型一致且是基本類型,值相等的時候,才能==,非基本類型會panic panic: runtime er
ror: comparing uncomparable type []int

2、== 適用的類型

我們日常開發(fā)中,經(jīng)常見到使用==的類型一般是:string,int等基本類型。struct有時候可以用有時候不可以。slice和map使用 ==會報錯.


int1:=10
int2:=10
str1:="11"
str2:="11"
if int1 == int2{}
if str1 == str2{}

int和string是值類型,我們直接對比他們的值。當然,前提是類型要一致,類型不一致編譯過不了。

3、slice和map使用 ==

首先golang里面有種說法:


切片之間不允許比較。切片只能與nil值比較。
map之間不允許比較。map只能與nil值比較。

那么我們分別測試下發(fā)現(xiàn):


(1)map比較會報錯:map can only be compared to nil
(2)切片報錯:the operator == is not defined on []int64
    slice can only be compared to nil

(1)那么兩個nil是否可以==比較呢

答案是不能:invalid operation: nil == nil (operator == not defined on nil)

(2)slice,map使用==的場景

就像上面說的,slice和map只能和nil使用==,他們各自之間是不可以的。


s1 := []int64{1, 2}
if s1 == nil {} //編輯器不會提示報錯

(3)為什么slice和map不可以

因為slice和map不止是需要比較值,還需要比較len和cap,層級比較深的話還需要遞歸比較,不是簡單的==就可以比較的,具體的我們可以參照reflect.DeepEqual()中實現(xiàn)的切片對比代碼。另外有大佬也說會出現(xiàn)循環(huán)引用的問題。

4、channel使用 ==

channel是引用類型,對比的是存儲數(shù)據(jù)的地址。channel是可以使用==的,只要類型一樣就可以。


ch1 := make(chan int, 1)
ch2 := ch1
if cha2 == cha1{fmt.Println("true")}

5、struct結構體使用==

(1)首先要明確幾點:


1)結構體的定義只是一種內存布局的描述,只有當結構體實例化時,才會真正地分配內存。
實例化就是根據(jù)結構體定義的格式創(chuàng)建一份與格式一致的內存區(qū)域,結構體實例與實例間的
內存是完全獨立的
2)對結構體進行&取地址操作時,視為對該類型進行一次 new 的實例化操作
因此:go中的結構體: v = Struct {}, v = &Struct{} 這個兩種寫法是等價的

結構體這里比較復雜一些。我們可以先下結論:

1、簡單結構的結構體,里面都是值類型或者指針的話,是可以使用 ==的
2、結構體中含有slice或者map,都是不可以用==

(2)簡單結構體的==


type Value struct {
    Name   string
    Gender string
    }
 
    func main() {
        v1 := Value{Name: "test", Gender: "男"}
        v2 := Value{Name: "test", Gender: "男"}
        if v1 == v2 {
            fmt.Println("true")
            return
        }
    }

(3)帶指針的結構體==

    首先要明確,指針類型指向的地址不一樣,肯定是沒辦法比較的,如果地址一樣,那么也可以用==
    首先要明確,指針類型指向的地址不一樣,肯定是沒辦法比較的,如果地址一樣,那么也可以用==
    type Value struct {
    Name   string
    Gender *string
    }
 
    func main() {
         Gender :=new(string) //下面賦值用的同一個變量,地址相同
        v1 := Value{Name: "test", Gender: Gender}
        v2 := Value{Name: "test", Gender: Gender}
        if v1 == v2 {
            fmt.Println("true")
            return
        }
    }


(4)強制轉換類型的==


type StructA struct {
    Name string
    }
 
    type StructB struct {
        Name string
    }
 
    func main() {
        s1 := StructA{Name: "test1"}
        s2 := StructB{Name: "test1"}
        if s1 == StructA(s2) {
            fmt.Println("true")
            return
        }    
   }

那復雜類型的結構體呢,要如何對比相等?包括slice和map如何對比相等呢?接下里就引入其他的對比方式。

三、reflect.DeepEqual() 和cmp.Equal()

1、reflect.DeepEqual()

reflect包提供的深度對比(遞歸)的方法,適用于go中的slice,map,struct,function的對比。

(1)對比規(guī)則


    相同類型的值是深度相等的,不同類型的值永遠不會深度相等。
    當數(shù)組值(array)的對應元素深度相等時,數(shù)組值是深度相等的。
    當結構體(struct)值如果其對應的字段(包括導出和未導出的字段)都是深度相等的,則該值是深度相等的。
    當函數(shù)(func)值如果都是零,則是深度相等;否則就不是深度相等。
    當接口(interface)值如果持有深度相等的具體值,則深度相等。
    當切片(slice)序號相同,如果值,指針都相等,那么就是深度相等的
    當哈希表(map)相同的key,如果值,指針都相等,那么就是深度相等的。

(2)對比實例

通過規(guī)則可以知道,reflect.DeepEqual是可以比較struct的,同時也可以用來比較slice和map。
     func main() {
        s1 := StructA{Name: "test", Hobby: []string{"唱", "跳"}}
        s2 := StructA{Name: "test", Hobby: []string{"唱", "跳"}}
        if reflect.DeepEqual(s1, s2) {
            fmt.Println("struct true")
        }
        mp1 := map[int]int{1: 10, 2: 20}
  mp2 := map[int]int{1: 10, 2: 20}
         if ok := reflect.DeepEqual(mp1, mp2);ok {
    fmt.Println("mp1 == mp2!")
      } else {
    fmt.Println("mp1 != mp2!")
      }
    }
  

2、cmp.Equal()

go-cmp是Google開源的比較庫,它提供了豐富的選項。

這個包旨在成為reflect.DeepEqual比較兩個值在語義上是否相等的更強大和更安全的替代方案。

參考:在測試中使用go-cmp

(1)對比規(guī)則:


1.在經(jīng)過路徑過濾,值過濾和類型過濾之后,會生一些忽略、轉換、比較選項,如果選項中存在忽略,
則忽略比較,如果轉換器比較器的數(shù)據(jù)大于1,則會panic(因為比較操作不明確)。如果選項中
存在轉換器,則調用轉換器轉換當前值,再遞歸調用轉換器輸出類型的Equal。如果包含一個比較器。
則比較使用比較器比較當前值。否則進入下一比較階段。


2.如果比較值有一個(T) Equal(T) bool 或者 (T) Equal(I) bool,那么,即使x與y是nil,
也會調用x.Equal(y)做為結果。如果不存在這樣的方法,則進入下一階段。


3.在最后階段,Equal方法嘗試比較x與y的基本類型。使用go語言的 == 比較基本類型
(bool, intX, float32,float64, complex32,complex64, string, chan)。


4.在比較struct時,將遞歸的比較struct的字段。如果結構體包含未導出的字段,函數(shù)會panic。
可以通過指定cmpopts.IgnoreUnexported來忽略未導出的字段,也可以使用
cmp.AllowUnexported來指定比較未導出的字段。

(2)代碼


// 傳入要對比的結構即可
func Equal(x, y interface{}, opts ...Option) bool {
  s := newState(opts)
  s.compareAny(rootStep(x, y))
  return s.result.Equal()
}


//獲取diff差異
func Diff(x, y interface{}, opts ...Option) string {
}


3、cmp和DeepEqual的區(qū)別?


安全:cmp.Equal()函數(shù)不會比較未導出字段(即字段名首字母小寫的字段)。遇到未導出字段,
cmp.Equal()直接panic,reflect.DeepEqual()會比較未導出的字段。


強大:cmp.Equal()函數(shù)提供豐富的函數(shù)參數(shù),讓我們可以實現(xiàn):忽略部分字段,比較零值,
轉換某些值,允許誤差等。


共同點:兩種比較類型,都會比較:對象類型,值,指針地址等。切片會按照索引比較值,
map是按照key相等比較值

四、其他對比方式

1、testify庫的 assert.Equal()

參考:Go 每日一庫之 testify

我們在寫單測的時候,經(jīng)常會使用assert.Equal()來做對比,原理如下:

(1)[]byte類型就使用bytes.Equal()
(2)非[]byte類型,使用reflect.DeepEqual()
(3)不相等則獲取diff

2、bytes.Equal()

標準庫中針對特定類型進行比較相等性的函數(shù)或方法。例如:想比較兩個byte slice就可以使用bytes.Compare函數(shù)。

reflect.DeepEqual()函數(shù)在比對slice的時候,如果發(fā)現(xiàn)是uint8類型,也就是[]byte類型,也會調用bytes包的對比方法


case Slice:
        // Special case for []byte, which is common.
        if v1.Type().Elem().Kind() == Uint8 {
         return bytealg.Equal(v1.Bytes(), v2.Bytes())
        }
        //轉化為string之間的對比
        func Equal(a, b []byte) bool {
         return string(a) == string(b)
        }


五、效率對比及總結

1、效率對比

效率對比參考:Golang幾種對象比較方法

1、簡單類型的==對比速度最快
2、復雜類型,自己知道結構之后寫的自定義對比速度次之
3、復雜結構且不確定結構的,使用cmp.Equal()或者reflect.DeepEqual()都可以,就是效率低點
4、assert.Equal()底層使用的就是reflect.DeepEqual()

2、總結

我們發(fā)現(xiàn)對比的兩個結構是否相等,方式很多,效率也有高有低。選擇合適自己需求的最重要。相對來說,cmp包是要更安全且可操作性更強一點,主要是看大家的喜好了。

鏈接:https://juejin.cn/post/7316450414158200847






審核編輯:劉清

聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權轉載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報投訴
  • 轉換器
    +關注

    關注

    27

    文章

    8602

    瀏覽量

    146720
  • 比較器
    +關注

    關注

    14

    文章

    1630

    瀏覽量

    107046
  • 存儲數(shù)據(jù)

    關注

    0

    文章

    85

    瀏覽量

    14083

原文標題:golang中如何比較struct,slice,map是否相等以及幾種對比方法的區(qū)別

文章出處:【微信號:magedu-Linux,微信公眾號:馬哥Linux運維】歡迎添加關注!文章轉載請注明出處。

收藏 人收藏

    評論

    相關推薦

    請教Xilinx Slice使用的問題

    在使用ISE進行實現(xiàn)過程中,占用slice的資源較多,如圖中所示,想用unrelated logic部分,期望能夠將slice資源均衡使用,而折騰了好久,改了好多綜合、map等的設置,可都
    發(fā)表于 02-28 14:55

    是否有指示MAP方法

    寄存器上的SRL。是否有指示MAP方法,它不應該在某些信號上推斷SRL?謝謝杰夫以上來自于谷歌翻譯以下為原文I have a design that has high clock rates.I
    發(fā)表于 10-10 10:52

    labview中二值化結果顯示的幾種方式對比

    不同的顏色顯示到image上。對比方法②和方法③的效果一樣,但是方法②更簡便,推薦使用方法②龍哥手把手教您LabVIEW視覺設計課程火熱上線?。≡斍榭牲c擊下方鏈接進行查看:http:/
    發(fā)表于 08-16 18:19

    XILINX MIG(DDR3) IP的AXI接口與APP接口的區(qū)別以及優(yōu)缺點對比

    XILINX MIG(DDR3) IP的AXI接口與APP接口的區(qū)別以及優(yōu)缺點對比
    發(fā)表于 11-24 21:47

    Delphi中比較GUID是否相等

    Delphi中比較GUID是否相等,CompareMem(@guid1, @guid2, SizeOf(TGUID)),最開始時想到的方法
    發(fā)表于 06-22 10:11 ?1686次閱讀

    幾種芯片的對比

    幾種芯片的對比,,希望有幫助對大家
    發(fā)表于 01-20 17:31 ?2次下載

    幾種步進電機加減速方法對比研究及其應用

    幾種步進電機加減速方法對比研究及其應用。
    發(fā)表于 05-03 11:44 ?12次下載

    typedef struct的用法

    typedef是類型定義的意思。typedef struct 是為了使用這個結構體方便。具體區(qū)別在于:若struct node{ }這樣來定義結構體的話。在定義 node 的結構體變量時,需要這樣寫:
    發(fā)表于 11-09 17:20 ?3342次閱讀

    Java Map幾種循環(huán)方式學習總結

    本文檔內容介紹了基于Java Map幾種循環(huán)方式學習總結,供參考
    發(fā)表于 03-19 15:51 ?0次下載

    分析對比幾種常用軸修復方法

    分析對比幾種常用軸修復方法
    發(fā)表于 12-02 11:05 ?1次下載

    C++中struct和class的區(qū)別?

    C++中struct和class的區(qū)別是什么?C++中struct和class的最大區(qū)別在于: ? ? ? ? struct的成員默認是公有
    的頭像 發(fā)表于 03-10 17:41 ?732次閱讀

    PostgreSQL準確且快速的數(shù)據(jù)對比方法

    NineData 數(shù)據(jù)對比是一款云原生數(shù)據(jù)對比產(chǎn)品,具備每秒處理 100 萬筆記錄的高效能力。它提供了一站式支持,適用于 IDC 自建、云主機自建以及云數(shù)據(jù)庫。NineData 支持多種數(shù)據(jù)庫,包括
    的頭像 發(fā)表于 09-12 15:46 ?931次閱讀
    PostgreSQL準確且快速的數(shù)據(jù)<b class='flag-5'>對比方法</b>

    List 轉 Map方法

    ; // 構造函數(shù) 、 get 、 set } 我們假定 id 字段 是唯一的, 所以我們把 id 作為 Map 的key。 使用 Java 8 之前的方法 在使用Java 8 之前,就只能使用比
    的頭像 發(fā)表于 10-09 16:10 ?1522次閱讀

    typedef struct和直接struct區(qū)別

    在C語言中, typedef 和 struct 是兩種不同的關鍵字,它們在定義和使用上有著明顯的區(qū)別。 typedef struct 和直接 struct 在 C 語言中用于定義結構體
    的頭像 發(fā)表于 08-20 10:58 ?2251次閱讀

    typedef和struct有啥區(qū)別

    在C語言中, typedef 和 struct 是兩個非常重要的關鍵字,它們在定義數(shù)據(jù)結構時扮演著關鍵的角色。然而,它們之間有一些明顯的區(qū)別。 1. struct 關鍵字 struct
    的頭像 發(fā)表于 08-20 11:00 ?840次閱讀