0

0

Go语言中Map存储结构体值与指针的差异与选择

霞舞

霞舞

发布时间:2025-07-16 13:10:14

|

839人浏览过

|

来源于php中文网

原创

go语言中map存储结构体值与指针的差异与选择

本文深入探讨了Go语言中map[int]struct和map[int]*struct两种存储结构体方式的根本区别。前者存储结构体的副本,对原始结构体的修改不会影响map中的值,且map中取出的值不可直接修改其成员。后者存储结构体的指针,对原始结构体或通过map取出的指针进行修改,会直接影响map中的值。文章通过示例代码详细解释了这两种方式在内存管理、数据修改行为上的差异,并提供了选择建议,帮助开发者理解何时选择值类型何时选择指针类型以优化程序行为。

在Go语言中,当我们需要将结构体作为Map的值存储时,通常会面临两种选择:存储结构体的值类型(map[KeyType]StructType)或存储结构体的指针类型(map[KeyType]*StructType)。这两种方式在内存管理、数据修改行为以及程序性能上存在显著差异。理解这些差异对于编写高效且可预测的Go程序至关重要。

值类型存储:复制与独立性

当Map的值类型是结构体本身(例如map[int]vertex)时,每次向Map中添加结构体或从Map中取出结构体时,Go都会对该结构体进行一次复制。这意味着Map中存储的是结构体的一个独立副本,而非其原始引用。

考虑以下示例代码:

package main

import "fmt"

type vertex struct {
    x, y int
}

func main() {
    a := make(map[int]vertex)  // Map存储vertex值类型
    b := make(map[int]*vertex) // Map存储vertex指针类型

    v := &vertex{0, 0} // 声明一个vertex指针
    a[0] = *v          // 将v指向的结构体值复制一份存入a[0]
    b[0] = v           // 将v的指针地址存入b[0]

    // 改变原始指针v指向的结构体内容
    v.x, v.y = 4, 4
    fmt.Println("修改原始v后:", a[0].x, a[0].y, b[0].x, b[0].y)

    // 尝试直接修改map中值类型结构体的成员(会导致编译错误)
    // a[0].x = 3 // 编译错误: cannot assign to (a[0]).x
    // a[0].y = 3 // 编译错误: cannot assign to (a[0]).y

    // 修改map中指针类型结构体的成员
    b[0].x = 3
    b[0].y = 3
    fmt.Println("修改b[0]后:", a[0].x, a[0].y, b[0].x, b[0].y)

    // 从map中取出值并修改
    u1 := a[0] // u1是a[0]的一个副本
    u1.x = 2
    u1.y = 2
    u2 := b[0] // u2是b[0]指向的指针的副本(即指向同一个内存地址)
    u2.x = 2
    u2.y = 2
    fmt.Println("修改u1/u2后:", a[0].x, a[0].y, b[0].x, b[0].y)
}

运行上述代码,输出如下:

火山写作
火山写作

字节跳动推出的中英文AI写作、语法纠错、智能润色工具,是一款集成创作、润色、纠错、改写、翻译等能力的中英文 AI 写作助手。

下载

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

修改原始v后: 0 0 4 4
修改b[0]后: 0 0 3 3
修改u1/u2后: 0 0 2 2

分析 map[int]vertex 的行为:

  1. *`a[0] = v**: 当*v(即{0, 0}这个vertex值)被赋值给a[0]时,Map内部会创建一个vertex结构体的副本,并存储这个副本。此时a[0]的值为{0, 0}`。
  2. v.x, v.y = 4, 4: 随后对原始指针 v 所指向的结构体进行修改,将其变为 {4, 4}。由于 a[0] 存储的是 v 在赋值时的副本,因此 a[0] 的值不会受到影响,依然是 {0, 0}。这在第一次 fmt.Println 输出中得到了验证:a[0].x, a[0].y 仍为 0 0。
  3. a[0].x = 3 编译错误: 这是因为从Map中取出的值类型结构体(a[0])是一个不可寻址的临时副本。Go语言不允许直接修改不可寻址值的字段。如果你想修改 a[0] 的内容,你必须先取出它,修改其副本,然后再将修改后的副本重新赋值回Map:
    tempVertex := a[0] // 取出a[0]的副本
    tempVertex.x = 3   // 修改副本
    a[0] = tempVertex  // 将修改后的副本重新存回map
  4. u1 := a[0]: 当 a[0] 赋值给 u1 时,同样会发生一次复制。u1 成为 a[0] 的一个独立副本。因此,对 u1 的修改(u1.x = 2, u1.y = 2)不会影响 a[0] 在Map中的原始值。在第三次 fmt.Println 输出中,a[0].x, a[0].y 仍然是 0 0。

指针类型存储:引用与共享

当Map的值类型是结构体的指针(例如map[int]*vertex)时,Map中存储的是

相关专题

更多
golang结构体相关大全
golang结构体相关大全

本专题整合了golang结构体相关大全,想了解更多内容,请阅读专题下面的文章。

193

2025.06.09

golang结构体方法
golang结构体方法

本专题整合了golang结构体相关内容,请阅读专题下面的文章了解更多。

184

2025.07.04

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

311

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

510

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

46

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

177

2025.08.29

Go中Type关键字的用法
Go中Type关键字的用法

Go中Type关键字的用法有定义新的类型别名或者创建新的结构体类型。本专题为大家提供Go相关的文章、下载、课程内容,供大家免费下载体验。

233

2023.09.06

go怎么实现链表
go怎么实现链表

go通过定义一个节点结构体、定义一个链表结构体、定义一些方法来操作链表、实现一个方法来删除链表中的一个节点和实现一个方法来打印链表中的所有节点的方法实现链表。

441

2023.09.25

苹果官网入口直接访问
苹果官网入口直接访问

苹果官网直接访问入口是https://www.apple.com/cn/,该页面具备0.8秒首屏渲染、HTTP/3与Brotli加速、WebP+AVIF双格式图片、免登录浏览全参数等特性。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

10

2025.12.24

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Rust 教程
Rust 教程

共28课时 | 3.8万人学习

Kotlin 教程
Kotlin 教程

共23课时 | 2万人学习

Go 教程
Go 教程

共32课时 | 2.9万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号