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部分

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)
}




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])
}