本篇文章基于 Golang 1.17.2
又到了金三银四的季节,作为一年一跳槽的胖虎,又准备开始跳槽了,这不他今天又请假来面试了。
“跳跳虎” 对于今天的面试还是有点准备的,八股文早就从网上下载好了,但就是来得及背完,就接到面试邀请了。胖虎心想,“不用怂,人称八股文选手不是白得的,要是今天能唬住面试官就多要点。”
纷争开始了
面试官:“写过C/C++的同学都知道,调用著名的malloc和new函数可以在堆上分配一块内存,这块内存的使用和销毁的责任都在程序员。一不小心,就会发生内存泄漏。那你说下Golang 是怎么处理这个问题的”
胖虎:“Golang 通过逃逸分析,对内存管理进行的优化和简化,它可以决定一个变量是分配到堆上还是栈上。”
什么是golang的逃逸分析
面试官:“那你说下什么是逃逸分析吧”
胖虎想:“这道题我会啊,准备好了吗,我要开始装X了。”
Golang 的逃逸分析,是指编译器根据代码的特征和生命周期,自动把变量分配到堆或者是栈上面。
通过优化了内存管理机制,解放广大程序员的双手。让程序员更关注于业务。
注意:Go 在编译阶段确立逃逸,并不是在运行时。
什么是栈与堆
面试官:“那你说下什么是栈和堆”
胖虎心:“这个也简单啊”
栈( stack)是系统自动分配空间的,例如我们定义一个 char a;系统会自动在栈上为其开辟空间。
而堆(heap)则是程序员根据需要自己申请的空间,例如 malloc(10);开辟十个字节的空间。
先看下内存分配图
栈在内存中是从高地址向下分配的,并且连续的,遵循先进后出原则。系统在分配的时候已经确定好了栈的大小空间。
栈上面的空间是自动回收的,所以栈上面的数据的生命周期在函数结束后,就被释放掉了。
堆分配是从低地址向高地址分配的,每次分配的内存大小可能不一致,导致了空间是不连续的,这也产生内存碎片的原因。由于是程序分配,所以效率相对慢些。
而堆上的数据只要程序不释放空间,就一直可以访问到,不过缺点是一旦忘记释放会造成内存泄露
逃逸分析有什么好处
面试官:“那你说下逃逸分析有什么好处吗”
胖虎:“你是十万个为什么吗?”, 但胖虎还是拿出了自己的看家本领。
就像刚开始提到的,Go 语言中内存的分配不是由程序员自己决定的,而是通过编译阶段确定的分配到何处。这样有什么好处呢?
没错机智的你,可能已经猜到了就是为了优化程序,榨干机器性能,让内存能够得到更高的使用效率。
通过逃逸分析,那些不需要分配到堆上的变量直接分配到栈上,堆上的变量少了不但同时减少 GC 的压力,还减轻了内存分配的开销。
常见的逃逸现象
面试官点点头,称赞的眼光看着胖虎说:“那你再说说,常见的逃逸现象有哪些吧”
胖虎内心崩溃了:“就我一个人一直在这说,都要渴死了,倒是给我来杯水啊,能不能让我喘口气”。但一想 JD 上面给的薪资还是挺诱惑人的,看在 的份上算了吧。
func(函数类型)数据类型
执行如下代码:
-m:表示内存分析 -l:表示防止内联优化
结果如下,第11行变量 name 逃逸到了堆上
interface{} 数据类型
同理执行逃逸分析,结果如下, name 变量也逃逸到堆上了
原因是 go/src/fmt/print.go 文件中 Println 方法传参数类型 interface{}, 编译器对传入的变量类型未知,所有统一处理分配到了堆上面去了。
指针 数据类型
结果如下,第11行变量 name 逃逸到了堆上
面试官:还有其他情况出现变量逃逸吗?
“额,这……”,胖虎心想:时间太匆忙了,八股文我就背了这么点啊,其他的还么来得及看呢,要是昨天少玩一把游戏就好了。这可怎么办?
看着胖虎憋的满脸通红,面试官笑呵呵的说,没事的,时间也不早了,今天先到这吧,你还有什么要问我的吗?
胖虎:还有哪些会出现变量逃逸啊
面试官:channel 或者栈空间不足逃逸, 也会导致逃逸的情况,还有其他问题吗?
胖虎:你们双休吗
面试官:是的,我们双休,确切来说是一个月休息两天
胖虎:“这。。。"
你们说胖虎还要继续面试下去吗?这样的公司能入职吗?
关注公众号:后端时光,发送“Golang”免费获取学习资料,没有套路,只有真情!
除了书籍之外,还有各互联网大厂最新面试题,超多经典电子书都可以免费获取哦~
我是 @后端时光,一个不务正业、不愿躺平的年轻人