golang指针和值类型在内存中的存储方式

值类型直接存储数据,赋值时复制独立副本;指针存储地址,共享同一数据,修改相互影响;内存分配由逃逸分析决定,上自动释放,堆上由GC管理。

golang指针和值类型在内存中的存储方式

在 Go 语言中,变量的存储方式取决于其类型是值类型还是指针类型,以及变量是在栈上分配还是可能逃逸到堆上。理解指针和值类型在内存中的存储方式,有助于写出更高效、更安全的代码。

值类型的内存存储

Go 中的大多数基础类型(如 int、float64、bool、struct、array)都是值类型。当声明一个值类型的变量时,它的实际数据会被直接存储在分配的内存空间中。

例如:

type Person struct {    Name string    Age  int}var p1 Person = Person{Name: "Alice", Age: 25}

变量 p1 是一个值类型的结构体实例,它的字段 Name 和 Age 的值会直接存储在该变量对应的内存块中。如果这个变量是在函数内定义的且没有逃逸,它通常会被分配在栈上;否则可能分配在堆上,由垃圾回收器管理。

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

赋值操作会复制整个值:

p2 := p1  // 复制 p1 的所有字段到 p2p2.Name = "Bob"

这时 p1.Name 仍然是 “Alice”,因为 p2 拥有独立的副本。

指针类型的内存存储

指针变量存储的是另一个变量的内存地址。声明一个指针类型时,它本身占用一小块内存(通常是 8 字节,在 64 位系统上),用于存放目标变量的地址。

例如:

p := &p1  // p 是 *Person 类型,存储 p1 的地址

变量 p 是一个指针,它保存的是 p1 在内存中的地址。通过 *p 可以访问或修改 p1 的内容。

多个指针可以指向同一个内存地址:

p3 := p  // p3 和 p 指向同一个 Person 实例p3.Age = 30

此时 p1.Age 也会变成 30,因为它们共享同一块数据。

栈与堆的分配机制

Go 编译器会进行逃逸分析,决定变量是分配在栈上还是堆上。

  • 局部值类型变量通常分配在栈上,函数返回后自动释放。
  • 如果一个局部变量被外部引用(比如返回其指针),就会逃逸到堆上,由 GC 管理生命周期。
  • 使用 new 或 make 创建的对象也通常分配在堆上。

例如:

func newPerson() *Person {    p := Person{Name: "Tom", Age: 20}  // 栈上创建    return &p  // p 逃逸到堆}

尽管 p 是值类型,但由于返回了它的地址,编译器会将其分配在堆上,确保指针指向的内存不会随函数结束而失效。

总结:值 vs 指针的内存行为

值类型:数据直接存储在变量所在内存中,赋值即复制,各自独立。 指针类型:变量存储的是地址,多个指针可共享同一数据,修改影响所有引用。 内存位置(栈或堆)由逃逸分析决定,开发者无需手动控制,但需理解其影响。 大结构体建议用指针传参,避免不必要的复制开销。

基本上就这些。Go 的设计让内存管理对开发者透明,但了解底层存储方式能帮助你写出更高效的程序。不复杂但容易忽略。

以上就是golang指针和值类型在内存中的存储方式的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1415729.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月16日 09:53:06
下一篇 2025年12月16日 09:53:11

相关推荐

发表回复

登录后才能评论
关注微信