go 变量逃逸分析
0. 前言
在 小白学标准库之 reflect 篇中介绍了反射的三大法则以及变量的逃逸分析。对于逃逸分析的介绍不多,大部分都是引自 Go 逃逸分析。不过后来看反射源码的过程中发现有一种情况 Go 逃逸分析 没讲透。且当时没从底层汇编的角度去看,导致有种似懂非懂的感觉。这里就变量逃逸内容进行介绍。
1. 逃逸分析案例
这里的案例不同于 Go 逃逸分析,当然所属情况是其中概括的几种类型。
1.1 全局变量和局部变量
示例代码:
var a int
func main() {
x := 10
a = x
println(a, x)
}
代码非常简单,定义全局变量 a 和局部变量 x,然后调用 println 打印 a 和 x。
使用 go tool compile 查看编译情况,注意变量逃逸在编译阶段,而不是运行时确定的。所以这里用 go tool compile 是能确定变量逃逸情况的:
// -m 查看变量逃逸情况
$ go tool compile -m escape.go
escape.go:5:6: can inline main
escape.go:39:6: can inline escapes
escape.go:39:14: leaking param: x
// 使用 -l 关闭函数内联
$ go tool compile -m -l escape.go
escape.go:39:14: leaking param: x
打印 leaking param: x 表明 x 代码中并未对 x 做任何引用操作,x 是一个泄露参数。不过,对于变量逃逸分析不影响,从结果来看,全局变量和局部变量都是在栈上分配的。
进一步思考,为什么全局变量会在栈上分配呢?
因为对全局变量赋值是传值的,传值就意味着这个值不是原有值,是值的拷贝。所以原有值不需要逃逸到堆上,只需要在栈上做变量拷贝就行。
改写示例代码如下:
var a *int
func main() {
x := 10
a = &x
println(a, x)
}
将全局变量改为全局指针类型变量,指针指向局部变量 x。查看变量逃逸情况:
$ go tool compile -l -m escape.go
escape.go:7:2: moved to heap: x
可以看到,变量 x 被 moved 到堆中。不难理解,全局变量指向 x,如果 x 不移到堆中,当 x 释放时,其它函数通过全局变量 a 找不到 x 了。事实上这是 c/c++ 语言会出现的情况。
通过汇编代码也能验证这点:
$ go tool compile -N -S -l escape.go
...
CALL runtime.newobject(SB)
继续改写上述代码:
func main() {
x := 10
a := &x
println(a, x)
}
这里用一个局部指针类型变量 a 指向 x,查看变量分配情况:
$ go tool compile -l -m escape.go
可以看到,变量 x 和 a 都是在栈上分配的。编译器检查到 a 是个指针类型变量并不会被外部作用域引用,可以将 x 放在栈上分配。
1.2 interface{} 型变量逃逸
在 go 接口学习笔记 中介绍了接口类型的表示。
对于 interface{} 类型的运行时表示为 runtime.eface:
type eface struct {
_type *_type
data unsafe.Pointer
}
这是空接口的运行时表示,对于编译阶段用于反射的空接口表示是 reflect.emptyInterface:
type emptyInterface struct {
typ *rtype
word unsafe.Pointer
}
知道了 interface{} 的反射表示,我们看示例代码:
func main() {
var a int = 10
var ai interface{} = a
println(ai)
}
逃逸分析:
$ go tool compile -l -m escape.go
改写示例代码:
func main() {
var a int = 10
var ai interface{} = &a
println(ai)
}
逃逸分析:
$ go tool compile -l -m escape.go
可以看到,对于局部变量 interface{} 类型变量转换,不管是赋值还是赴地址都没有变量逃逸。这里发生了什么其实和上一节的局部变量一样,就不过多分析了。
值得提的一点是,给 interface{} 传地址,结构体的 word 将指向地址,而给 interface{} 传值,结构体的 word 是一个指针,将指向值所在的内存地址。这里由于是局部变量,这个变量值 a 是在栈上分配的,结构体 word 指向的是栈上值所在的地址。
再改写示例代码 1:
var ai interface{}
func main() {
var a int = 10
ai = a
println(ai)
}
逃逸分析:
$ go tool compile -l -m escape.go
escape.go:9:5: a escapes to heap
示例代码 2:
var ai interface{}
func main() {
var a int = 10
ai = &a
println(ai)
}
逃逸分析:
$ go tool compile -l -m escape.go
escape.go:8:6: moved to heap: a
可以看到,对于全局变量 ai 不管是传值还是传地址,变量 a 都将逃逸到堆中。为什么会这样也好理解:interface{} 反射的结构体表示是指针 data: unsafe.Pointer。
通过汇编代码看传值的例子:
$ go tool compile -N -S -l escape.go
CALL runtime.convT64(SB)
重点看 runtime.convT64 函数,该函数会在堆上分配内存。详细看这里,不在展开了。
1.3 反射
示例代码如下:
func main() {
var a int = 10
var ai interface{} = a
fmt.Println(ai)
}
逃逸分析:
$ go tool compile -l -m escape.go
escape.go:11:13: ... argument does not escape
escape.go:11:13: a escapes to heap
这里代码除了 fmt.Println 改动基本和 1.2 节代码一样,为什么这时候 a 就逃到堆上了呢?
原因肯定在于 fmt.Println 函数,查看函数我们发现代码会走到 escapes(i) 这里,escapes 的函数实现是:
// Dummy annotation marking that the value x escapes,
// for use in cases where the reflect code is so clever that
// the compiler cannot follow.
func escapes(x interface{}) {
if dummy.b {
dummy.x = x
}
}
var dummy struct {
b bool
x interface{}
}
再解释这段实现之前,先看下为什么要用 escapes(i) 函数:
// TODO: Maybe allow contents of a Value to live on the stack.
// For now we make the contents always escape to the heap. It
// makes life easier in a few places (see chanrecv/mapassign
// comment below).
comment:
Note: some of the noescape annotations below are technically a lie,
but safe in the context of this package. Functions like chansend
and mapassign don't escape the referent, but may escape anything
the referent points to (they do shallow copies of the referent).
It is safe in this package because the referent may only point
to something a Value may point to, and that is always in the
heap (due to the escapes() call in ValueOf).
说白了,不用 escapes() 会让编译器很麻烦,这里涉及到 noescape,详细了解可看这里。
escapes 实际上是一种欺骗行为,欺骗编译器使得编译器将变量逃逸到堆中。怎么欺骗的呢?其实和结合上两节分析,基本能看出来了。
在变量 escapes(i) 到 escapes(x interface{}) 时发生了类型转换,将 i 转换为 interface{} 类型,实际做的就是 1.2 节描述的行为。然后,重点在 dummy.x == x,全局变量 dummy.x 会引用转换的接口 x,由于 dummy.x 是一个 interface{} 类型,其实质是一个指针,所以编译器会将 interface x 中 data 指向的变量 i 分配到堆中。这里注意 i 可以是值也可以是地址,如果是地址,编译器会将地址指向的值分配到堆中。
可能描述起来较为复杂,复杂的原因是 interface{} 做了好几层包装。我们拆开包装,用一种简化方式看代码的欺骗行为:
var a *int
func main() {
x := 10
var f bool
if f {
a = &x
}
}
逃逸分析:
$ go tool compile -l -m escape.go
escape.go:33:2: moved to heap: x
可以看到,骗过了编译器使得变量 x 逃逸到了堆上,虽然 a = &x 不会执行。
1.4 总结
本篇文章通过几个逃逸分析案例重点分析 escapes 函数是如果做到欺骗编译器实现变量逃逸的。
go 变量逃逸分析的更多相关文章
- Go变量逃逸分析
目录 什么是逃逸分析 为什么要逃逸分析 逃逸分析是怎么完成的 逃逸分析实例 总结 写过C/C++的同学都知道,调用著名的malloc和new函数可以在堆上分配一块内存,这块内存的使用和销毁的责任都在程 ...
- JVM中启用逃逸分析
-XX:+DoEscapeAnalysis 逃逸分析优化JVM原理我们知道java对象是在堆里分配的,在调用栈中,只保存了对象的指针.当对象不再使用后,需要依靠GC来遍历引用树并回收内存,如果对象数量 ...
- JVM笔记-逃逸分析
参考: http://www.iteye.com/topic/473355http://blog.sina.com.cn/s/blog_4b6047bc01000avq.html 什么是逃逸分析(Es ...
- JVM逃逸分析
开启逃逸分析: -server -XX:+DoEscapeAnalysis -XX:+PrintGCDetail -Xmx10m -Xms10m 关闭逃逸分析: -server -XX:-DoEsca ...
- 逃逸分析(Escape Analysis)
一.什么是逃逸 逃逸是指在某个方法之内创建的对象,除了在方法体之内被引用之外,还在方法体之外被其它变量引用到:这样带来的后果是在该方法执行完毕之后,该方法中创建的对象将无法被GC回收,由于其被其它变量 ...
- golang逃逸分析和竞争检测
最近在线上发现一块代码逻辑在执行N次耗时波动很大1ms~800ms,最开始以为是gc的问题,对代码进行逃逸分析,看哪些变量被分配到堆上了,后来发现是并发编程时对一个切片并发的写,导致存在竞争,类似下面 ...
- 深入理解Java中的逃逸分析
在Java的编译体系中,一个Java的源代码文件变成计算机可执行的机器指令的过程中,需要经过两段编译,第一段是把.java文件转换成.class文件.第二段编译是把.class转换成机器指令的过程. ...
- JVM的逃逸分析
我们都知道Java中的对象默认都是分配到堆上,在调用栈中,只保存了对象的指针.当对象不再使用后,需要依靠GC来遍历引用树并回收内存.如果堆中对象数量太多,回收对象还有整理内存,都会会带来时间上的消耗, ...
- Java之JVM逃逸分析
引言: 逃逸分析(Escape Analysis)是众多JVM技术中的一个使用不多的技术点,本文将通过一个实例来分析其使用场景. 概念 逃逸分析,是一种可以有效减少Java 程序中同步负载和内存堆分配 ...
- java虚拟机的逃逸分析
逃逸分析作为其他优化手段提供依据的分析技术,其基本行为就是分析对象动态作用域:当一个对象在方法中被定义后,它可能被外部方法所引用,例如作为调用参数传递到其他方法中,称为方法逃逸.甚至还有可能被外部线程 ...
随机推荐
- Mongodb安装篇+可视化工具篇
下载MongoDB 官网下载地址:Download MongoDB Community Server | MongoDB Version 选择:稳定版4.4.2 Mongo的版本分为稳定版和开发版 ...
- Redis入门实践
安装Redis 下载:官网:https://redis.io/download/,选择稳定版下载. 上传至linux 解压Redis:tar -zxvf redis-6.2.7.tar.gz,得到: ...
- lxml模块
lxml主要用xpath.css选择器等来提取xml格式文档,html也是xml格式文档的一种. xpath方法返回列表的三种情况 返回空列表:没有找到任何元素 返回字符串列表:xpath规则匹配用了 ...
- 三维GIS引擎用什么好?结合目前市面上的主流引擎进行分析
相信大多数人在谈到三维GIS引擎时,第一个想到的首先是CesiumJS,CesiumJS以其免费开源的特点,快速占领了三维GIS这个领域,同时也催生了许多以CesiumJS为基础的衍生产品.Cesiu ...
- ElasticSearch之Shard request cache settings
对于查询操作,Elasticsearch提供了缓存特性来暂存结果. 对于相同条件的查询请求,在缓存中的数据失效前,响应后续的查询操作时可以直接从缓存中提取结果,有效降低检索操作的时延,提升检索数据时的 ...
- uniapp-welive仿微信/抖音直播带货|uni-app+vue3+pinia短视频直播商城
基于uniapp+vue3+uv-ui跨端H5+小程序+App短视频+直播带货商城Uniapp-WeLive. uni-welive一款全新基于uniapp+vue3+pinia+vk-uview等技 ...
- 部署堡垒机2——安装MySQL8.0.32或8.0.28+
MySQL的三大版本 a)MySQL Enterprise Edition:企业版本(付费)b)MySQL Cluster CGE:高级集群版(收费)c)MySQL Community Server: ...
- 十分钟教你在 k8s 中部署一个前后端应用
转载至我的博客https://www.infrastack.cn ,公众号:架构成长指南 大家好,我是蜗牛哥,好多开发人员,尤其是没接触过 k8s 的人员对如何在k8s中部署一个 前后端应用很模糊,不 ...
- openGauss数据库在CentOS上的安装实践
本文分享自华为云社区<openGauss数据库在CentOS上的安装实践>,作者:Gauss小松鼠 . 1.安装前准备 安装数据库前先要有已安装centOS 7.6的服务器+数据库安装包. ...
- 突破开源Redis的内存限制,存算分离的GaussDB到底有多能“装”?
摘要:GaussDB(for Redis)(下文简称高斯Redis)是华为云数据库团队自主研发的兼容Redis协议的云原生数据库,该数据库采用计算存储分离架构,突破开源Redis的内存限制,可轻松扩展 ...