golang中的 for range 结合 & 取地址的趣味小问题

Posted 李斌的BLOG

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了golang中的 for range 结合 & 取地址的趣味小问题相关的知识,希望对你有一定的参考价值。

前言

在网上发现一道golang 中的 for range& 取地址结合的题目,感觉很有趣,先来看看演示代码。

演示代码

package main

import "fmt"

type student struct {
	name string
	age  int
}

func main() {
	m := make(map[string]*student)
	stus := []student{
		{name: "小王子", age: 18},
		{name: "娜扎", age: 23},
		{name: "大王八", age: 9000},
	}

	for _, stu := range stus {
		m[stu.name] = &stu
	}
	for k, v := range m {
		fmt.Println(k, "=>", v.name)
	}
}

输出结果:

小王子 => 大王八
娜扎 => 大王八
大王八 => 大王八

是不是很惊愕,结果为什么不是预期的,map的值都是一个呢?

原因分析

for range 每次产生的 keyvalue 其实是对应的 stus 里面值的拷贝,不是对应的 stus 里面的值的引用,所以出现了这种问题。

stustusfor循环中申请的一个局部变量,每次循环都会拷贝 stus 中对应的值 stu。迭代遍历之后,stu 每次会被重新赋值,而在 m 这个 map 中记录的 value 只不过是 stu内存地址

我可以打印一下 &stu 的内存地址:

... ...

for _, stu := range stus {
		fmt.Printf("%p\\n",&stu)
		m[stu.name] = &stu
	}

... ...

输出内存地址如下:

0xc00000c030
0xc00000c030
0xc00000c030
小王子 => 大王八
娜扎 => 大王八
大王八 => 大王八

会发现所有的内存地址都是一样的

解决方案

重新申请一个变量,即可解决

... ...

for _, stu := range stus {
		s := stu
		m[stu.name] = &s
	}

... ...

输出内存地址如下:

大王八 => 大王八
小王子 => 小王子
娜扎 => 娜扎

注意:

修改后,这里输出的结果每次运行结果都不一样,因为 map 数据是无序的。

以上是关于golang中的 for range 结合 & 取地址的趣味小问题的主要内容,如果未能解决你的问题,请参考以下文章

Golang 语言坑之for-range

为啥正常的 for 循环允许为结构字段分配值,而 for range 在 Golang 中不起作用? [复制]

Go语言自学系列 | golang for range循环

golang range for循环中如何正确的给goroutine传参

Golang 基础:Go Module, for range, 切片, map, struct 等使用和实现

Golang 基础:Go Module, for range, 切片, map, struct 等使用和实现