golang 结构体作为map的元素时,不能够直接赋值给结构体的某个字段

golang 结构体作为map的元素时,不能够直接赋值给结构体的某个字段

 

 

引入: 错误 Reports assignments directly to a struct field of a map

golang 结构体作为map的元素时,不能够直接赋值给结构体的某个字段

1. 问题的产生

这个问题在github上可以追溯到2012年提交的一个issue,链接为https://github.com/golang/go/issues/3117;如上图,结构体作为map的元素时,不能够直接赋值给结构体的某个字段,也就是map中的struct中的字段不能够直接寻址。

2. 问题产生的原因

关于golang中map的这种古怪的特性有这样几个观点:

1)map作为一个封装好的数据结构,由于它底层可能会由于数据扩张而进行迁移,所以拒绝直接寻址,避免产生野指针;

2)map中的key在不存在的时候,赋值语句其实会进行新的k-v值的插入,所以拒绝直接寻址结构体内的字段,以防结构体不存在的时候可能造成的错误;

3)这可能和map的并发不安全性相关

3. 问题的解决

1)迂回方式一:整体更新map的value部分

 
  1. package main

  2.  
  3. import "fmt"

  4.  
  5. type Person struct{

  6. name string

  7. sex string

  8. age int

  9. }

  10.  
  11. func main(){

  12. m := map[uint]Person{

  13. 0 : Person{"张无忌", "男", 18},

  14. 1 : Person{"周芷若", "女", 17},

  15. }

  16.  
  17. //m[0].age += 1

  18. //整体更新结构体

  19. temp := m[0]

  20. temp.age += 1

  21. m[0] = temp

  22. fmt.Println(m)

  23. }

运行结果:

golang 结构体作为map的元素时,不能够直接赋值给结构体的某个字段

2) 迂回方式二:把map的value部分定义为对应类型的指针类型或是slice或是map时,这样是可以更新v的内部字段的

 
  1. package main

  2.  
  3. import "fmt"

  4.  
  5. type Person struct{

  6. name string

  7. sex string

  8. age int

  9. }

  10.  
  11. func main() {

  12. //定义map的value类型为指针类型

  13. m := map[uint]*Person{

  14. 0: &Person{"张无忌", "男", 18},

  15. 1: &Person{"周芷若", "女", 17},

  16. }

  17.  
  18. m[0].age += 1

  19.  
  20. fmt.Println(*m[0])

  21. }

运行结果:

golang 结构体作为map的元素时,不能够直接赋值给结构体的某个字段