Go语言中struct、array、slice和map值的比较

2023-06-01 00:00:00 struct 语言 Array

1.可以使用相等运算符 == 来比较结构体变量,前提是两个结构体的成员都是可比较的类型:

type data struct {
    num     int
    fp      float32
    complex complex64
    str     string
    char    rune
    yes     bool
    events  <-chan string
    handler interface{}
    ref     *byte
    raw     [10]byte
}

func main() {
    v1 := data{}
    v2 := data{}
    fmt.Println("v1 == v2: ", v1 == v2)    // true
}


如果两个结构体中有任意成员是不可比较的,将会造成编译错误。注意数组成员只有在数组元素可比较时候才可比较。


type data struct {
    num    int
    checks [10]func() bool        // 无法比较
    doIt   func() bool        // 无法比较
    m      map[string]string    // 无法比较
    bytes  []byte            // 无法比较
}

func main() {
    v1 := data{}
    v2 := data{}
    fmt.Println("v1 == v2: ", v1 == v2)
}
invalid operation: v1 == v2 (struct containing [10]func() bool cannot be compared)


2.Go 提供了一些库函数来比较那些无法使用 == 比较的变量,比如使用 "reflect" 包的 DeepEqual():


// 比较相等运算符无法比较的元素
func main() {
    v1 := data{}
    v2 := data{}
   
    fmt.Println("v1 == v2: ", reflect.DeepEqual(v1, v2))    // true
   
    m1 := map[string]string{"one": "a", "two": "b"}
    m2 := map[string]string{"two": "b", "one": "a"}
   
    fmt.Println("v1 == v2: ", reflect.DeepEqual(m1, m2))    // true
   
    s1 := []int{1, 2, 3}
    s2 := []int{1, 2, 3}
   
       // 注意两个 slice 相等,值和顺序必须一致
    fmt.Println("v1 == v2: ", reflect.DeepEqual(s1, s2))    // true
   
}


3.这种比较方式可能比较慢,根据你的程序需求来使用。DeepEqual() 还有其他用法:


func main() {
    var b1 []byte = nil
   
    b2 := []byte{}
   
    fmt.Println("b1 == b2: ", reflect.DeepEqual(b1, b2))    // false
   
}


注意两点:

一.DeepEqual() 并不总适合于比较 slice


func main() {
    var str = "one"
    var in interface{} = "one"
    fmt.Println("str == in: ", reflect.DeepEqual(str, in))    // true
   
    v1 := []string{"one", "two"}
    v2 := []string{"two", "one"}
   
    fmt.Println("v1 == v2: ", reflect.DeepEqual(v1, v2))    // false
   
    data := map[string]interface{}{
        "code":  200,
        "value": []string{"one", "two"},
    }
   
    encoded, _ := json.Marshal(data)
    var decoded map[string]interface{}
    json.Unmarshal(encoded, &decoded)
   
    fmt.Println("data == decoded: ", reflect.DeepEqual(data, decoded))    // false
}



如果要大小写不敏感来比较 byte 或 string 中的英文文本,可以使用 “bytes” 或 “strings” 包的 ToUpper () 和 ToLower () 函数。比较其他语言的 byte 或 string,应使用 bytes.EqualFold () 和 strings.EqualFold ()


如果 byte slice 中含有验证用户身份的数据(密文哈希、token 等),不应再使用 reflect.DeepEqual ()、bytes.Equal ()、 bytes.Compare ()。这三个函数容易对程序造成 timing attacks,此时应使用 “crypto/subtle” 包中的 subtle.ConstantTimeCompare () 等函数


二.reflect.DeepEqual () 认为空 slice 与 nil slice 并不相等,但注意 byte.Equal () 会认为二者相等:


func main() {
    var b1 []byte = nil
    b2 := []byte{}
   
    // b1 与 b2 长度相等、有相同的字节序
    // nil 与 slice 在字节上是相同的
    fmt.Println("b1 == b2: ", bytes.Equal(b1, b2))    // true
   
}


更多go语言的注意点,请收藏本站网址:https://www.zongscan.com/



相关文章