Go struct 类型的 map 结构体成员不能修改的问题

Darker_坤 · · 4204 次点击 · · 开始浏览    
这是一个创建于 的文章,其中的信息可能已经有所发展或是发生改变。

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

引入: 错误 Reports assignments directly to a struct field of a 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的并发不安全性相关

  • x = y 这种赋值的方式,你必须知道 x的地址,然后才能把值 y 赋给 x。
  • 但 go 中的 map 的 value 本身是不可寻址的,因为 map 的扩容的时候,可能要做 key/val pair迁移
  • value 本身地址是会改变的
  • 不支持寻址的话又怎么能赋值呢

3. 问题的解决

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

package main
 
import "fmt"
 
type Person struct{
    name string
    sex string
    age int
}
 
func main(){
    m := map[uint]Person{
        0 : Person{"张无忌", "男", 18},
        1 : Person{"周芷若", "女", 17},
    }
 
    //m[0].age += 1
    //整体更新结构体
    temp := m[0]
    temp.age += 1
    m[0] = temp
    fmt.Println(m)
}

运行结果:

image

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

package main
 
import "fmt"
 
type Person struct{
    name string
    sex string
    age int
}
 
func main() {
    //定义map的value类型为指针类型
    m := map[uint]*Person{
        0: &Person{"张无忌", "男", 18},
        1: &Person{"周芷若", "女", 17},
    }
 
    m[0].age += 1
 
    fmt.Println(*m[0])
}

运行结果:

image

有疑问加站长微信联系(非本文作者)

本文来自:简书

感谢作者:Darker_坤

查看原文:Go struct 类型的 map 结构体成员不能修改的问题

入群交流(和以上内容无关):加入Go大咖交流群,或添加微信:liuxiaoyan-s 备注:入群;或加QQ群:692541889

4204 次点击  
加入收藏 微博
暂无回复
添加一条新回复 (您需要 登录 后才能回复 没有账号 ?)
  • 请尽量让自己的回复能够对别人有帮助
  • 支持 Markdown 格式, **粗体**、~~删除线~~、`单行代码`
  • 支持 @ 本站用户;支持表情(输入 : 提示),见 Emoji cheat sheet
  • 图片支持拖拽、截图粘贴等方式上传