如何理解Go指针和内存分配关系_Go Pointer内存Model解析
技术百科
P粉602998670
发布时间:2026-01-01
浏览: 次 Go指针变量分配位置取决于作用域和逃逸分析,而非指针本身;其指向的数据位置由创建方式决定;逃逸分析是编译期自动判断变量是否需堆分配的关键机制。
Go指针本身不特殊,它就是一个存地址的变量,大小固定(64位系统是8字节),分配位置取决于作用域和逃逸分析,而不是“因为是指针所以放堆上”。真正影响性能和行为的,是它指向的数据在哪、生命周期多长。
指针变量存在哪?看作用域和逃逸
指针变量和其他变量一样,遵循Go统一的内存分配规则:
- 函数内声明的局部指针,比如 var p *int,只要没被返回、没存进全局变量或堆对象,就分配在栈上
- 一旦编译器发现这个指针的值(即它指向的地址)会被函数外使用,比如 return &x,那 x 就会逃逸到堆,p 本身也可能被移到堆上(取决于是否被外部引用)
- 包级指针变量(如 var globalP *string)直接放在
数据段,属于静态区,程序启动时就存在,结束时才释放
指针指向的数据在哪?看创建方式
指针的值是地址,这个地址指向的内容,由创建它的方式决定:
- &x:x 本身在哪,地址就指向哪。x 在栈 → 指向栈;x 因逃逸在堆 → 指向堆
- new(T) 或 &T{}:明确在堆上分配内存,返回的指针一定指向堆
- 切片/Map/Channel 的元素地址:底层数据总在堆上,所以取到的指针也指向堆
- 栈上的指针完全可以指向堆数据(很常见),堆上的指针也可以指向其他堆数据(比如结构体字段是指针)
逃逸分析才是关键裁判
Go不靠程序员手动指定栈或堆,而是由编译器在编译期做逃逸分析,判断变量是否“逃出”当前作用域:
- 如果一个局部变量的地址被返回、赋给全局变量、传入 goroutine 或存进 map/slice,它就必须堆分配
- 命令 go build -gcflags="-m -l" 可以查看逃逸详情,比如 “moved to heap” 或 “escapes to heap”
- 逃逸不是坏事,是安全机制:避免函数返回后访问已销毁的栈内存
值类型 vs 指针类型:别只看大小
选值传递还是指针传递,不能只看结构体大不大:
- 小结构体(如两个 int 字段)值传递开销极小,栈上拷贝快,GC 无压力
- 大结构体用指针可避免拷贝,但每个指针本身带来 8 字节开销,且若指向堆,会增加 GC 扫描负担
- 更关键的是语义需求:需要修改原值?要实现某个接口(如 io.Writer)?方法接收者是否统一?这些常比内存数字更重要
基本上就这些。理解指针和内存的关系,核心是分清“指针变量在哪”和“它指向的数据在哪”,再把逃逸分析当作背后的自动调度员——它不动声色,但决定了大部分实际布局。
# 的是
# 就会
# 放在
# 是一个
# 才是
# 更重要
# 它就
# 是由
# go
# 对象
# 堆
# String
# int
# 值类型
# 字节
# 指针
# 接口
# 栈
# pointer
# var
# 结构体
# 作用域
# 切片
# map
# channel
# 指针类型
# 局部变量
# 全局变量
# 值传递
# 只看
相关栏目:
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
AI推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
SEO优化<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
技术百科<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
谷歌推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
百度推广<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
网络营销<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
案例网站<?muma echo $count; ?>
】
<?muma
$count = M('archives')->where(['typeid'=>$field['id']])->count();
?>
【
精选文章<?muma echo $count; ?>
】
相关推荐
- Win11怎么设置系统还原_Windows11系统
- c# 在高并发下使用反射发射(Reflection
- Win10怎样安装Word样式库_Win10安装W
- C++如何编写函数模板?(泛型编程入门)
- Win11怎么更改计算机名_Windows11系统
- Windows执行文件被SmartScreen拦截
- 如何在 Laravel 中通过嵌套关联关系进行 o
- 如何用::实现单例模式_php静态方法与作用域操作
- Go语言中slice追加操作的底层共享机制详解
- Windows10如何更改开机密码_Win10登录
- 如何提升Golang程序I/O性能_Golang
- ACF 教程:正确更新嵌套在多层 Group 字段
- Win11怎么设置虚拟内存_Windows 11优
- Windows11怎么用“记事本”自动换行与编码
- Win10怎样清理C盘爱奇艺缓存_Win10清理爱
- Win11如何卸载OneDrive_Win11卸载
- php怎么下载安装后设置默认字符集_utf8配置步
- Win10如何卸载自带Edge_Win10彻底卸载
- Win11如何隐藏桌面图标 Win11一键隐藏/显
- Win11怎么恢复出厂设置_Win11重置此电脑保
- 新手学PHP架构总混淆概念咋办_重点梳理【教程】
- C++友元类使用场景_C++类间协作设计方式讲解
- Windows10如何更改鼠标图标_Win10鼠标
- Win11怎么关闭自动维护 Win11禁用系统自动
- Mac如何调整Dock栏大小和位置_Mac程序坞个
- Python大文件处理策略_内存优化说明【指导】
- Win11怎么更改盘符_Win11磁盘管理修改驱动
- C++ static_cast和dynamic_c
- Win11怎么关闭定位服务 Win11禁止应用获取
- 如何使用Golang搭建本地API测试环境_快速验
- win11如何清理传递优化文件 Win11为C盘瘦
- Python项目回滚策略_发布安全说明【指导】
- 如何使用Golang实现RPC序列化与反序列化_G
- Python函数接口文档化_自动化说明【指导】
- 如何在JavaScript中动态拼接PHP的bas
- php485在macos下怎么配置_php485
- Win11怎么退出微软账户_切换Win11为本地账
- Mac如何彻底清理浏览器缓存?(Safari与Ch
- Win10系统怎么查看网络连接状态_Windows
- XSLT怎么生成动态的HTML属性名和标签名
- Windows10系统怎么查看IP地址_Win10
- Win11怎么关闭用户账户控制UAC_Window
- Win11视频默认播放器怎么改_Win11关联第三
- 如何在 Go 中创建包含 map 的 slice(
- Win11截图快捷键是什么_Win11自带截图工具
- Win11怎么开启空间音效_Windows11耳机
- VSC怎样在VSC中调试PHPAPI_接口调试技巧
- php打包exe后无法读取环境变量_变量配置方法【
- Win10怎么关闭自动更新错误重启 Win10策略
- C++中引用和指针有什么区别?(代码说明)

数据段,属于静态区,程序启动时就存在,结束时才释放
QQ客服