检查一个条件下多个映射中是否存在键。

huangapple go评论114阅读模式
英文:

Check if key exists in multiple maps in one condition

问题

我需要检查两个映射中是否存在相同的键:

  1. if v1, ok1 := map1["aaa"]; ok1 {
  2. ...
  3. }
  4. if v2, ok2 := map2["aaa"]; ok2 {
  5. ...
  6. }

是否可能将这两个条件合并为一个?我设法做到了这样:

  1. v1, ok1 := map1["aaa"]
  2. v2, ok2 := map2["aaa"]
  3. if ok1 && ok2 {
  4. ...
  5. }

但我想知道是否可以在一个 if 条件中完成赋值和检查。

英文:

I need to check if the same key exists in two maps:

  1. if v1, ok1 := map1["aaa"]; ok1 {
  2. ...
  3. }
  4. if v2, ok2 := map2["aaa"]; ok2 {
  5. ...
  6. }

Is it possible to join these two conditions into one? I managed to do something like this:

  1. v1, ok1 := map1["aaa"]
  2. v2, ok2 := map2["aaa"]
  3. if ok1 && ok2 {
  4. ...
  5. }

but I'm curious whether it (assigning and checking) can be done in one if condition.

答案1

得分: 8

不,这是不可能的。规范:索引表达式:

在特殊形式的赋值或初始化中,对类型为map[K]V的映射a进行索引表达式操作,例如:

  1. v, ok = a[x]
  2. v, ok := a[x]
  3. var v, ok = a[x]

会产生一个额外的无类型布尔值。如果键x存在于映射中,ok的值为true,否则为false

因此,只有在没有其他赋值操作时,才能使用特殊的v, ok := m[k]形式。

然而,如果你不使用映射值类型的零值,你可以使用简单的元组赋值来进行检查;通过不使用特殊形式,而是使用两个简单的索引表达式。

例如,如果你的值类型是某个接口类型(例如interface{}),并且你知道你不使用nil值,你可以这样做:

  1. if v1, v2 := m1["aaa"], m2["aaa"]; v1 != nil && v2 != nil {
  2. fmt.Printf("Both map contains key 'aaa': %v, %v\n", v1, v2)
  3. }

当然,通过一个辅助函数,你可以一步完成:

  1. func idx(m1, m2 map[string]interface{}, k string) (
  2. v1, v2 interface{}, ok1, ok2 bool) {
  3. v1, ok1 = m1[k]
  4. v2, ok2 = m2[k]
  5. return
  6. }

使用它:

  1. if v1, v2, ok1, ok2 := idx(m1, m2, "aaa"); ok1 && ok2 {
  2. fmt.Printf("Both map contains key 'aaa': %v, %v\n", v1, v2)
  3. }

Go Playground上尝试这些示例。

英文:

No, it can't be done. Spec: Index expressions:

> An index expression on a map a of type map[K]V used in an assignment or initialization of the special form
>
> v, ok = a[x]
> v, ok := a[x]
> var v, ok = a[x]
>
> yields an additional untyped boolean value. The value of ok is true if the key x is present in the map, and false otherwise.

So you can use the special v, ok := m[k] form only if nothing else gets assigned.

However, if you don't use the zero value of the value type of the map, you can do the check using a simple tuple-assignment; by not using the special form but 2 simple index expressions.

For example if your value type is some interface type (e.g. interface{}), and you know you don't use the nil value, you may do the following:

  1. if v1, v2 := m1["aaa"], m2["aaa"]; v1 != nil && v2 != nil {
  2. fmt.Printf("Both map contains key '%s': %v, %v\n", "aaa", v1, v2)
  3. }

Of course with a helper function, you can do it in one step:

  1. func idx(m1, m2 map[string]interface{}, k string) (
  2. v1, v2 interface{}, ok1, ok2 bool) {
  3. v1, ok1 = m1[k]
  4. v2, ok2 = m2[k]
  5. return
  6. }

Using it:

  1. if v1, v2, ok1, ok2 := idx(m1, m2, "aaa"); ok1 && ok2 {
  2. fmt.Printf("Both map contains key '%s': %v, %v\n", "aaa", v1, v2)
  3. }

Try the examples on the Go Playground.

答案2

得分: 0

你也可以使用可变参数(三个点)来检查多个键:

  1. // 检查 map1 和 map2 是否为空
  2. func checkMap(m1, m2 map[string]interface{}, keys ...string) []bool {
  3. var isExist []bool
  4. for _, key := range keys {
  5. // 从 map1 检查值
  6. _, ok := m1[key]
  7. if ok {
  8. isExist = append(isExist, true) // 第一次追加以避免 panic
  9. } else {
  10. isExist = append(isExist, false) // 第一次追加以避免 panic
  11. }
  12. // 从 map2 检查值
  13. _, ok = m2[key]
  14. if ok {
  15. isExist[len(isExist)-1] = true
  16. } else {
  17. isExist[len(isExist)-1] = false
  18. }
  19. }
  20. return isExist
  21. }

然后你可以按顺序检查你的键,像这样:

  1. result := checkMap(myMap, myMap2, "a", "b", "c", "d", "e", "f", "g")
  2. fmt.Printf("result = %+v\n", result) // 打印结果
  3. if result[0] {
  4. fmt.Println("key a 在两个 map 中存在")
  5. }
  6. if result[1] {
  7. fmt.Println("key b 在两个 map 中存在")
  8. }
英文:

you could also use variadic parameter(three dots) to check multiple keys :

  1. // check map 1 and map2 is null or not
  2. func checkMap(m1, m2 map[string]interface{}, keys ...string) []bool {
  3. var isExist []bool
  4. for key := range keys {
  5. //checking value from map 1
  6. _, ok := m1[keys[key]]
  7. if ok {
  8. isExist = append(isExist, true) // append for the first time to avoid panic
  9. } else {
  10. isExist = append(isExist, false) // append for the first time to avoid panic
  11. }
  12. // checking value from map2
  13. _, ok = m2[keys[key]]
  14. if ok {
  15. isExist[key] = true
  16. } else {
  17. isExist[key] = false
  18. }
  19. }
  20. return isExist
  21. }

And then you can check your keys in order like this :

  1. result := checkMap(myMap, myMap2, "a", "b", "c", "d", "e", "f", "g")
  2. fmt.Printf("result = %+v\n", result) // print the result
  3. if result[0] {
  4. fmt.Println("key a exist in both map")
  5. }
  6. if result[1] {
  7. fmt.Println("key b exist in both map")
  8. }

huangapple
  • 本文由 发表于 2017年2月1日 19:01:53
  • 转载请务必保留本文链接:https://go.coder-hub.com/41978101.html
匿名

发表评论

匿名网友

:?: :razz: :sad: :evil: :!: :smile: :oops: :grin: :eek: :shock: :???: :cool: :lol: :mad: :twisted: :roll: :wink: :idea: :arrow: :neutral: :cry: :mrgreen:

确定