Go语言面试题:为什么for range循环中使用指针会导致所有值变成最后一个元素?

Go语言面试题:为什么for range循环中使用指针会导致所有值变成最后一个元素?

go语言面试题:深入理解for…range循环指针

Go语言的for…range循环简洁高效,但使用指针时容易引发误解。本文剖析一个常见的面试题,阐明for…range循环中指针的陷阱及解决方法

问题描述

考虑一个学生结构体和学生切片,目标是将学生信息存储到map中并打印姓名:

type student struct {     name string     age  int }  func main() {     m := make(map[string]*student)     stus := []student{         {name: "pprof.cn", age: 18},         {name: "测试", age: 23},         {name: "博客", age: 28},     }      for _, stu := range stus {         m[stu.name] = &stu     }     for k, v := range m {         fmt.Println(k, "=>", v.name)     } }

预期输出:

pprof.cn => pprof.cn 测试 => 测试 博客 => 博客

实际输出:

立即学习go语言免费学习笔记(深入)”;

pprof.cn => 博客 测试 => 博客 博客 => 博客

问题分析

所有v.name都变成了”博客”,原因在于for…range循环的迭代变量stu在每次迭代中是同一个变量,只是其值被更新。&stu创建的指针始终指向这个单一变量。循环结束后,stu的值为最后一个学生的信息,因此map中的所有指针都指向了这个最终的stu变量。

Go语言的for…range循环复用迭代变量,不会为每次迭代创建新变量。因此,获取迭代变量的地址将始终得到同一个地址。

解题方法

为了解决这个问题,需要在每次迭代中创建stu的副本:

for _, stu := range stus {     stuCopy := stu // 创建副本     m[stu.name] = &stuCopy }

通过创建stuCopy副本,并使用&stuCopy作为map的值,确保每个学生的信息都被独立存储和引用。

正确输出

修改后的代码将产生预期的输出:

pprof.cn => pprof.cn 测试 => 测试 博客 => 博客

总结

本例揭示了for…range循环中使用指针的潜在风险。 理解迭代变量的复用机制,并在必要时创建副本,是避免此类问题的关键。 这对于Go语言开发者来说是一个重要的知识点。

以上就是Go语言面试题:

© 版权声明
THE END
喜欢就支持一下吧
点赞10 分享