Go语言【学习】defer和逃逸分析
defer
什么是defer?
defer是Go语言的一中用于注册延迟调用的机制,使得函数活语句可以再当前函数执行完毕后执行
为什么需要defer?
Go语言提供的语法糖,减少资源泄漏的发生
如何使用defer?
在创建资源语句的附近,使用defer语句释放资源
示例一:
/*func f1()(r int){
t := 5
// 1.赋值指令
r = t
// defer在赋值和返回之间执行
defer func() {
t = t + 5
}()
// 空的return指令
return t
}*/ package main import "fmt" func f1()(r int){
t := 5
defer func() {
t = t + 5
}()
return t
} func main() {
fmt.Println(f1())
} //5
执行return指令时,首先会把返回值copy到栈上,返回空的IP指令;return时先执行了赋值操作r=t,后执行defer操作,最后r的值没有修改
示例二:
package main import "fmt" func f2()(r int){
defer func() {
r = r + 5
}()
return 1
} func main() {
fmt.Println(f2())
} //6
能懂示例一,这个自然懂
示例三:
package main import "fmt" func f3()(r int){
defer func(r *int) {
*r = *r + 5
}(&r)
return 1
} func main() {
fmt.Println(f3())
} //6
defer传入的是指针,修改会改变原来的值,所以依然是6
示例四:
package main import (
"errors"
"fmt"
) func e1(){
var err error
// 压栈的时候 err已经变成nil值
defer fmt.Println("e1",err)
err = errors.New("defer1 error")
fmt.Println(err)
return
} func e2(){
var err error
// 闭包err是外部err的引用
defer func() {
fmt.Println("e2",err)
}()
err = errors.New("defer2 error")
return
} func e3(){
var err error
// 参数拷贝时就是nil
defer func(err error) {
fmt.Println("e3",err)
}(err)
err = errors.New("defer3 error")
return
} func main() {
e1()
e2()
e3()
} //e1 <nil>
//e2 defer2 error
//e3 <nil>
示例五:
package main import "fmt" func main() {
var a = accumulator()
fmt.Println(a(1))
fmt.Println(a(10))
fmt.Println(a(100))
var b = accumulator()
fmt.Println(b(1))
fmt.Println(b(10))
fmt.Println(b(100))
} func accumulator() func(int) int{
var x int return func(i int) int {
fmt.Printf("(%+v, %+v) - ",&x,x)
x += i
return x
} } //(0xc00000a0b8, 0) - 1
//(0xc00000a0b8, 1) - 11
//(0xc00000a0b8, 11) - 111
//(0xc00000a120, 0) - 1
//(0xc00000a120, 1) - 11
//(0xc00000a120, 11) - 111
示例六(执行顺序):
package main import (
"fmt"
"time"
) func main() {
defer fmt.Println("defer main")
var user = "" go func() {
defer func() {
fmt.Println("defer caller")
if err := recover(); err != nil{
fmt.Println("recover success . err:", err)
}
}() func(){
defer func() {
fmt.Println("defer here")
}()
if user == ""{
panic("should set user env.")
}
}()
}()
time.Sleep(time.Second)
fmt.Println("end") }
示例七:
package main import "fmt" func main() { for i := 0; i < 5; i++ {
defer fmt.Println(i,1) } for i := 0; i < 5; i++ {
defer func(){
fmt.Println(i,2)
}()
} for i := 0; i < 5; i++ {
defer func(){
j := i
fmt.Println(j,3)
}()
} for i := 0; i < 5; i++ {
j := i
defer fmt.Println(j,4)
} for i := 0; i < 5; i++ {
j := i
defer func() {
fmt.Println(j,5)
}()
} // 拷贝传值
for i := 0; i < 5; i++ {
defer func(j int) {
fmt.Println(j, 6)
}(i)
}
} //4 6
//3 6
//2 6
//1 6
//0 6
//4 5
//3 5
//2 5
//1 5
//0 5
//4 4
//3 4
//2 4
//1 4
//0 4
//5 3
//5 3
//5 3
//5 3
//5 3
//5 2
//5 2
//5 2
//5 2
//5 2
//4 1
//3 1
//2 1
//1 1
//0 1
//
逃逸分析
什么是逃逸分析?
Go语言编译器执行静态代码分析后,决定哪些变量逃逸到堆上
为什么需要逃逸分析?
尽可能将变量分配到栈上
逃逸分析如何进行?
只有在编译器能证明变量在函数返回后不再被引用的,才会分配到栈上,其他情况分配到堆上
总结:
动态内存分配(堆上)比静态内存分配(栈上)开销要大的多
如果变量在函数外部没有引用,则优先放到栈中;如果在函数外部存在引用,则必定放到堆中;
示例一:
package main type S1 struct {} func main() {
var x S1
_ = indentity1(x)
} func indentity1(x S1) S1{
return x
}
逃逸检查
Z:\src\defer和逃逸分析>go build -gcflags "-m -l" escape1.go Z:\src\defer和逃逸分析>
没有逃逸,值传递,直接在栈上分配。Go语言函数传递都是通过值的,调用函数的时候,直接在栈上copy出一份参数,不存在逃逸
示例二:
package main type S2 struct {} func main() {
var x S2
y := &x
_ = indentity2(y)
} func indentity2(x *S2) *S2{
return x
}
逃逸检查
Z:\src\defer和逃逸分析>go build -gcflags "-m -l" escape2.go
# command-line-arguments
.\escape2.go:11:17: leaking param: x to result ~r1 level=0
.\escape2.go:7:7: main &x does not escape Z:\src\defer和逃逸分析>
x未发生逃逸,identity函数的输入直接当成返回值了,没有对x进行引用,所以x没有逃逸。
示例三:
package main type S3 struct {} func main() {
var x S3
_ = *ref3(x)
} func ref3(z S3) *S3{
return &z
}
逃逸检查
Z:\src\defer和逃逸分析>go build -gcflags "-m -l" escape3.go
# command-line-arguments
.\escape3.go:11:9: &z escapes to heap
.\escape3.go:10:11: moved to heap: z Z:\src\defer和逃逸分析>
示例四:
package main type S4 struct {
M *int
} func main() {
var i int
_ = ref4(i)
} func ref4(y int) (z S4){
z.M = &y
return z
逃逸检查
Z:\src\defer和逃逸分析>go build -gcflags "-m -l" escape4.go
# command-line-arguments
.\escape4.go:13:8: &y escapes to heap
.\escape4.go:12:11: moved to heap: y Z:\src\defer和逃逸分析>
示例五:
package main type S5 struct {
M *int
} func main() {
var i int
ref5(&i)
} func ref5(y *int) (z S5){
z.M = y
return z
}A
逃逸检查
Z:\src\defer和逃逸分析>go build -gcflags "-m -l" escape5.go
# command-line-arguments
.\escape5.go:12:11: leaking param: y to result z level=0
.\escape5.go:9:7: main &i does not escape Z:\src\defer和逃逸分析>
示例六:
package main type S6 struct {
M *int
} func main() {
var x S6
var i int
ref6(&i,&x)
} func ref6(y *int, z *S6){
z.M = y
}
逃逸检查
Z:\src\defer和逃逸分析>go build -gcflags "-m -l" escape6.go
# command-line-arguments
.\escape6.go:13:11: leaking param: y
.\escape6.go:13:19: ref6 z does not escape
.\escape6.go:10:7: &i escapes to heap
.\escape6.go:9:6: moved to heap: i
.\escape6.go:10:10: main &x does not escape
Go语言【学习】defer和逃逸分析的更多相关文章
- R语言学习笔记-Corrplot相关性分析
示例图像 首先安装需要的包 install.packages("Corrplot") #安装Corrplot install.packages("RColorBrewer ...
- 20155229-付钰涵-分析自我技能延展到c语言学习状况
我的小技能 我记得幼儿园时表演的舞蹈,也记得从水彩到素描的学习,还记得小学和初中获得的钢琴省级奖项. 舞蹈止于一年级,绘画止于三年级,钢琴从学前班到高一那十年的时间里有过断续. 03年-04年的那个冬 ...
- 足球运动训练心得及经验分析-c语言学习调查
在准备预备作业02之前,我参考娄老师的提示,阅读了<[做中学(Learning By Doing)]之乒乓球刻意训练一年总结>一文. 在文章描述的字里行间,给予我的印象是系统.负责,娄老师 ...
- Go 语言机制之逃逸分析
https://blog.csdn.net/weixin_38975685/article/details/79788254 Go 语言机制之逃逸分析 https://blog.csdn.net/ ...
- 基于Golang的逃逸分析(Language Mechanics On Escape Analysis)
何为逃逸分析 在编译程序优化理论中,逃逸分析是一种确定指针动态范围的方法——分析在程序的哪些地方可以访问到指针.它涉及到指针分析和形状分析. 当一个变量(或对象)在子程序中被分配时,一个指向变量的指针 ...
- 小师妹学JVM之:逃逸分析和TLAB
目录 简介 逃逸分析和栈上分配 TLAB简介 TLAB详解 设置TLAB空间的大小 TLAB中大对象的分配 TLAB空间中的浪费 总结 简介 逃逸分析我们在JDK14中JVM的性能优化一文中已经讲过了 ...
- 技能收获与C语言学习
你有什么技能比大多人(超过90%以上)更好? 我会的东西很多,喜欢的东西太多,但是很遗憾广而不专,会而不精.学了很多东西我都是为了娱乐,因为以前我们那里过于强调学习,很多爱好也都被扼杀在摇篮里.我觉得 ...
- go语言学习笔记
Go语言学习基本类型Bool 取值范围:true,false (不可以用数字代替)Int/uint 根据平台可能为32或64位int8/uint8 长度:1字节 取值范围-128~127/0~255b ...
- 20155306白皎 学习技能+C语言学习
你有什么技能比大多数人更好 谈起技能,我还有感觉有微微拿得出手的也只有主持这一项才艺了吧.从小学到高中一直参加朗诵比赛,以及从小学到大学一直在所在学校有担任过主持工作. 上大学以来,也参加了院级朗诵比 ...
随机推荐
- webpack资源管理
一.概况 ①webpack不仅可以打包JavaScript模块,甚至它把网页开发中的一切资源的都可以当作模块来打包处理 ②但是webpack本身不支持,它只是一个打包平台,其他资源,例如css.les ...
- bzoj 1260涂色 题解
题面 区间dp, 我学的也不怎么好. myj说动态规划就是搜索的无限剪枝. 所以是搜了网上的代码, 看了看. 思路就是枚举区间,f数组就是存储从i到j需要的最少次数, 当然一开始他们的值要先设置一个很 ...
- 【CF55D】Beautiful numbers
[CF55D]Beautiful numbers 题面 洛谷 题解 考虑到如果一个数整除所有数那么可以整除他们的\(lcm\),而如果数\(x\)满足\(x\bmod Lcm(1,2...,9)=r\ ...
- Macbook Pro升级10.15后百度网盘无法登陆,网络连接错误(-1001)
兴冲冲升级到10.15,结果百度网盘挂了~QQ 由于长期在境外实验室做研究,百度又封锁了境外登陆,所以客户端是唯一跟家里联络的方式,现在它也挂了感觉整个天都塌下来了. 找了一圈,发现一个特别神奇的解锁 ...
- PSQLException: An I/O error occurred while sending to the backend.
postgresql批量新增数据时,批量插入的数据量太大了,造成了IO异常 只能把这么多数据先进行分割,再批量插入,但是感觉这种方式不是最优解,先暂时顶着,具体List分割方式如下: package ...
- oracle 如何判断字符串中是否包含中文?超级简单!
1.情景展示 如何快速的判断出指定字符串中是否包含中文呢? 2.解决方案 通过length()和lengthb()函数的比对结果进行判断. lengthb(string)计算string所占的字节 ...
- sql注入用<>绕过被过滤的select ——百度杯9月第二场SQL
题目提示SQL注入,打开源代码发现<!--SELECT * FROM info WHERE id=1--> 尝试union select 1,2,3提示inj code! 经过多次尝试之后 ...
- 关于conda和jupyter使用
conda建立虚拟环境 常用指令 conda env list # 查看环境 conda create -n env_name python=3.6.8 # 安装对应python的虚拟环境 conda ...
- CentOS7下将Python的版本升级为3.7
众所周知,在2020年python官方将不再支持2.7版本的python,所以使用3.x版本的python是必要的,但是linux中默认安装的python一般是2.6和2.7版本的,要是使用linux ...
- Server Tomcat v8.5 Server at localhost was unable to start within 45 seconds. If the server requires more time, try increasing the timeout in the server editor.
Server Tomcat v9.0 Server at localhost was unable to start within 45 seconds. If the server requires ...