溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

Golang數(shù)據(jù)類型實(shí)例代碼比較分析

發(fā)布時(shí)間:2023-04-18 10:15:46 來源:億速云 閱讀:123 作者:iii 欄目:開發(fā)技術(shù)

這篇文章主要講解了“Golang數(shù)據(jù)類型實(shí)例代碼比較分析”,文中的講解內(nèi)容簡單清晰,易于學(xué)習(xí)與理解,下面請(qǐng)大家跟著小編的思路慢慢深入,一起來研究和學(xué)習(xí)“Golang數(shù)據(jù)類型實(shí)例代碼比較分析”吧!

分類說明是否能比較說明
基本類型整型( int/uint/int8/uint8/int16/uint16/int32/uint32/int64/uint64/byte/rune等)浮點(diǎn)數(shù)( float32/float64)復(fù)數(shù)類型( complex64/complex128)字符串( string)
引用類型切片(slice)、map
聚合類型(復(fù)合類型)數(shù)組相同長度的數(shù)組可以比較,不同長度的數(shù)組不能進(jìn)行比較

結(jié)構(gòu)體只包含可比較的類型情況下可比較
接口類型如error

 基本數(shù)據(jù)類

  • 類型一致且是基本類型,值相等的時(shí)候,才能==

  • 非基本類型會(huì)panic panic: runtime error: comparing uncomparable type []int

浮點(diǎn)比較

不過基本類型中也要注意浮點(diǎn)型的比較就不像我們現(xiàn)實(shí)中的一樣,比如0.1+0.2在計(jì)算中運(yùn)行結(jié)果就不是0.3了,而是0.30000000000000004了

package main import "fmt" func main() {     
    var a float64=0.1     
    var b float64=0.2     
    // 0.30000000000000004     
    fmt.Println(a+b) 
}

字符串比較

一般的比較運(yùn)算符(==、!=、<、<=、>=、>)是通過在內(nèi)存中按字節(jié)比較來實(shí)現(xiàn)字符串比較的,因此比較的結(jié)果是字符串自然編碼的順序。字符串所占的字節(jié)長度可以通過函數(shù) len() 來獲取,例如 len(str)

比較兩個(gè)字符是否相等

package golangbase

import (
	"fmt"
	"testing"
)

func TestString(t *testing.T) {
	str1 := "哈哈"
	str2 := "哈哈"
	fmt.Println(str1 == str2)
}

輸出結(jié)果為true

引用類型

slice、map

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

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

  • 兩個(gè)nil也不能比較,會(huì)panic

slice、map比較

使用reflect.DeepEqual()

對(duì)比規(guī)則

  • 相同類型的值是深度相等的,不同類型的值永遠(yuǎn)不會(huì)深度相等。

  • 當(dāng)數(shù)組值(array)的對(duì)應(yīng)元素深度相等時(shí),數(shù)組值是深度相等的。

  • 當(dāng)結(jié)構(gòu)體(struct)值如果其對(duì)應(yīng)的字段(包括導(dǎo)出和未導(dǎo)出的字段)都是深度相等的,則該值是深度相等的。

  • 當(dāng)函數(shù)(func)值如果都是零,則是深度相等;否則就不是深度相等。

  • 當(dāng)接口(interface)值如果持有深度相等的具體值,則深度相等。

  • 當(dāng)切片(slice)序號(hào)相同,如果值,指針都相等,那么就是深度相等的

  • 當(dāng)哈希表(map)相同的key,如果值,指針都相等,那么就是深度相等的。

使用示例

package golangbase

import (
	"reflect"
	"testing"
)

type StructA struct {
	Name  string
	Hobby []string
}

type StructB struct {
	Name string
}

func TestDeepEqual(t *testing.T) {
	s1 := StructA{Name: "test", Hobby: []string{"唱", "跳"}}
	s2 := StructA{Name: "test", Hobby: []string{"唱", "跳"}}
	println(reflect.DeepEqual(s1, s2))// true
	mp1 := map[int]int{1: 10, 2: 20}
	mp2 := map[int]int{1: 10, 2: 20}
	println(reflect.DeepEqual(mp1, mp2))// true
}

channel、指針

指針可比較,只要指針指向的地址一樣,則相等

由于通過make創(chuàng)建channel后,返回的是一個(gè)指針,所以可以比較

c1 := make(chan int, 2) 
 
c2 := make(chan int, 2) 
 
c3 := c1 
 
fmt.Println(c3 == c1) // true 
 
fmt.Println(c2 == c1) // false

聚合類型

數(shù)組

數(shù)組在go中是必須先確定長度的,也就是長度不能再去擴(kuò)容。并且它是個(gè)值拷貝,做參數(shù)傳到一個(gè)函數(shù)中被修改,那么外部的值還是一樣的不變的。Slice則相反。那么數(shù)組是否可以比較呢,看下面的例子:

package main
import "fmt"
func main() {
    a := [2]int{1, 2}
    b := [2]int{1, 2}
    c := [2]int{1, 3}
    d := [3]int{1, 2, 4}
    fmt.Println(a == b) // true
    fmt.Println(a == c) // false
    fmt.Println(a == d) // invalid operation: a == d (mismatched types [2]int and [3]int)
}

可以看出,相同長度的數(shù)組是可以比較的,而不同長度的數(shù)組是不能進(jìn)行比較的。原因是什么呢?這是因?yàn)閿?shù)組類型中,數(shù)組的長度也是類型的一部分,不同長度的數(shù)組那么他們的類型也就被認(rèn)為不同的,所以無法比較

結(jié)構(gòu)體

只包含可比較的類型情況下可比較

package main
import "fmt"
type A struct {
    id int
    name string
}
func main() {
    a := A{id:5,name:"123"}
    b := A{id:5,name:"123"}
    c := A{id:5,name:"1234"}
    fmt.Println(a == b) // true
    fmt.Println(a == c) // false
}

反例,因?yàn)閟lice不可比較,如果結(jié)構(gòu)體包含了slice,則不可比較

package main
import "fmt"
type A struct {
    id int
    name string
    son []int
}
func main() {
    a := A{id:5,name:"123",son:[]int{1,2,3}}
    b := A{id:5,name:"123",son:[]int{1,2,3}}
    fmt.Println(a == b) // invalid operation: a == b (struct containing []int cannot be compared)
}

接口

Go 語言根據(jù)接口類型是否包含一組方法將接口類型分成了兩類:

  • 使用 runtime.iface結(jié)構(gòu)體表示包含方法的接口

  • 使用 runtime.eface結(jié)構(gòu)體表示不包含任何方法的 interface{} 類型

type eface struct { // 16 字節(jié)
    _type *_type
    data  unsafe.Pointer
}

type iface struct { // 16 字節(jié)
    tab  *itab
    data unsafe.Pointer
}

一個(gè)接口值是由兩個(gè)部分組成的,即該接口對(duì)應(yīng)的類型和接口對(duì)應(yīng)具體的值

接口值的比較涉及這兩部分的比較,只有當(dāng)類型和值都相等(動(dòng)態(tài)值使用==比較),兩個(gè)接口值才是相等的。看個(gè)例子:

var a interface{} = 0
var b interface{} = 2
var c interface{} = 0
var d interface{} = 0.0
fmt.Println(a == b) // false
fmt.Println(a == c) // true
fmt.Println(a == d) // false

a和c類型相同(都是int),值也相同(都是0,基本類型比較),故兩者相等。 a和b類型相同,值不等,故兩者不等。 a和d類型不同,a為int,d為float64,故兩者不等

最后做個(gè)練習(xí)

func TestJson(t *testing.T) {
	var x, y Data
	x = Data{
		UUID:    "856f5555806443e98b7ed04c5a9d6a9a",
		Content: 1,
	}
	bytes, _ := json.Marshal(x)
	_ = json.Unmarshal(bytes, &y)
	println(x)
	println(y)
	println(reflect.DeepEqual(x, y))
}

為什么結(jié)果為false?

debug看一下

Golang數(shù)據(jù)類型實(shí)例代碼比較分析

原因是json.Unmarshal默認(rèn)會(huì)將所有的數(shù)字類型轉(zhuǎn)為float64

Golang數(shù)據(jù)類型實(shí)例代碼比較分析

針對(duì)這種情況,可以封裝一個(gè)DeepEqual方法

func DeepEqual(v1, v2 interface{}) bool {
	if reflect.DeepEqual(v1, v2) {
		return true
	}
	bytesA, _ := json.Marshal(v1)
	bytesB, _ := json.Marshal(v2)
	return bytes.Equal(bytesA, bytesB)
}

![在這里插入圖片描述](https://img-blog.csdnimg.cn/55d294d1ab7740aba7b547d1a6165b5a.png)

感謝各位的閱讀,以上就是“Golang數(shù)據(jù)類型實(shí)例代碼比較分析”的內(nèi)容了,經(jīng)過本文的學(xué)習(xí)后,相信大家對(duì)Golang數(shù)據(jù)類型實(shí)例代碼比較分析這一問題有了更深刻的體會(huì),具體使用情況還需要大家實(shí)踐驗(yàn)證。這里是億速云,小編將為大家推送更多相關(guān)知識(shí)點(diǎn)的文章,歡迎關(guān)注!

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請(qǐng)聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI