不破楼兰终不还——Go 延迟语句defer指南

说到defer,很多gopher都知道这是求职面试常考点,也是一个易错的难点,特别是延迟语句defer也是Golang一个十分重要的关键字。所以掌握defer刻不容缓!

什么是defer?

现在我们编程经常要操作文件或数据库,而进行数据库和文件操作就会涉及数据库和文件的关闭,用完不关闭就会导致内存泄露,可能会导致很严重的安全问题。但是这也是我们经常忘记的一个步骤,所以defer可以很好的解决这个问题,比如我们连接数据库就使用defer编写关闭语句,这就能较好的帮助开发人员编写更安全的程序。

不过defer运行时会带来一定时间的开销,因此如果对耗时要求特别严格,建议不使用defer。

我们来看下面这个例子:

var l sync.RWMutex
l.Lock()
panic("异常信息")
l.Unlock()

如果上面这种情况,使用锁和解锁语句之间出现了panic,就会形成死锁,即使我们不使用panic语句,其他语句也可能导致panic啊,因此我们需要使用defer。

defer的执行顺序是什么?

根据Golang官方文档描述,defer就像一个LIFO的栈,每次执行defer语句,都会将函数”压栈“,函数参数也会被保存下来;如果外层函数(非代码块)退出,最后的defer语句就会执行,也就是栈顶的函数或方法会被执行。

不过需要注意:

如果defer执行的语句是一个nil,那么就会在调用时产生panic。

一般情况下多个defer的执行顺序,我们可以通过下面这个例子了解:

package mainimport ("fmt"
)func main() {defer fmt.Println("defer 1")defer_test()defer fmt.Println("defer 2")
}func defer_test() {defer fmt.Println("defer 3")defer fmt.Println("defer 4")
}

运行结果为:

defer 4
defer 3
defer 2
defer 1Program exited.

defer就像一个LIFO的栈,最后被定义的defer最先执行。

参数传递无外乎就是传值(pass by value),传引用(pass by reference)或者说是传指针。在Go语言中,按引用传递其实也可以称作”按值传递”,只不过该副本是一个地址的拷贝,通过它可以修改这个值所指向的地址上的值。

使用defer时,涉及到函数参数和闭包引用。使用函数参数方式,defer会在定义时取值并保存起来。而使用闭包引用的方式,虽然也是值传递,但是拷贝的是函数指针。

举个栗子:

package mainimport ("fmt"
)func main() {var array [5]int = [5]int{5, 4, 3, 2, 1}for _, i := range array {defer func() { fmt.Println(i) }()}
}

运行结果如下:

1
1
1
1
1Program exited.

defer语句全是输出1,因为循环结束后i=1,而使用匿名函数让defer后面跟着的是一个“闭包”,所以i是“引用类型”的变量。

如果对上面这段代码稍作修改,得到的结果就不一样了:

package mainimport ("fmt"
)func main() {var array [5]int = [5]int{5, 4, 3, 2, 1}for _, i := range array {defer fmt.Println(i)}
}

运行结果如下:

1
2
3
4
5Program exited.

调用 defer 关键字会立刻拷贝函数中引用的外部参数,因此当i从5到1时,所有的值都被拷贝下来。

defer与return的执行顺序

defer用得好则已,用得不好就会带来灾难。

能不能用好就得看我们能不能理解retrun语句。

一条return语句,其实不是一条原子指令,其大概可以分为三条指令:

  • 返回值为xxx
  • 调用defer函数
  • 空的return

举个栗子:

package mainimport ("fmt"
)func main() {fmt.Println("return:", banana())
}func banana() (i int) {defer func() {i++fmt.Println("defer 2:", i) }()defer func() {i++fmt.Println("defer 1:", i) }()return i
}
defer 1: 1
defer 2: 2
return: 2Program exited.

这是有名返回值的情况,接下我们来看一看匿名返回值的情况:

package mainimport ("fmt"
)func main() {fmt.Println("return:", apple())
}func apple() int {var i intdefer func() {i++fmt.Println("defer 2:", i)}()defer func() {i++fmt.Println("defer 1:", i)}()return i
}
defer 1: 1
defer 2: 2
return: 0Program exited.

上面这两段代码说明了:defer语句只能访问有名返回值,不能直接访问匿名返回值。

但是如果是下面这种情况:

package mainimport ("fmt"
)func main() {fmt.Println("return:", banana())
}func banana() (i int) {defer func(i int) {i++fmt.Println("defer 2:", i)}(i)defer func(i int) {i++fmt.Println("defer 1:", i)}(i)return i
}

输出结果就为:

defer 1: 1
defer 2: 1
return: 0Program exited.

这是因为传递给defer后面的匿名函数的是形参的一个复制值,不会影响实参i。

参考文献

机械工业出版社 《Go程序员面试笔试宝典》

defer的执行顺序与时机 https://studygolang.com/articles/22931

理解 Go 语言 defer 关键字的原理 | Go 语言设计与实现

https://draveness.me/golang/docs/part2-foundation/ch05-keyword/golang-defer/#531-现象

GO函数传参 []int 与 [3]int 有何区别?https://segmentfault.com/q/1010000020543158?bd_source_light=4746641

Golang中defer、return、返回值之间执行顺序的坑 https://cloud.tencent.com/developer/article/1410243

不破楼兰终不还——Go 延迟语句defer指南相关推荐

  1. GO语言教程4:defer(延迟语句)详解

    文章目录 1.defer语句格式 2.defer执行的时间 3.defer语句的作用 4.defer语句执行的顺序 5.defer与return的value之间的关系 6.用defer进行代码跟踪 7 ...

  2. Go基础编程:延迟调用defer

    链客,专为开发者而生,有问必答! 此文章来自区块链技术社区,未经允许拒绝转载. 本篇文章所讲的就是go编程中的延迟调用defer,希望对社区的成员有较多的帮助. 1 defer作用 关键字defer ...

  3. golang for语句完全指南

    golang for语句完全指南 Posted on January 13, 2018 以下所有观点都是个人愚见,有不同建议或补充的的欢迎emialaboutme 原文章地址 关于for语句的疑问 f ...

  4. Go 延迟调用 defer 用法详解

    引子 package counterimport ("log""sync" )type Counter struct {mu *sync.MutexValue ...

  5. Verilog赋值间延迟语句与赋值内延迟语句比较

    module full_adder(a,b,sum); input a,b; output reg sum; always @(a,b) #13 sum = (a & b) ;   或者   ...

  6. 百战终破黄金甲,不破楼兰终不还!

    今天是值得纪念的一天,我的博客成功入选推荐博客,哈哈! 好开心!记得自己的第一篇博客成为推荐的时候我跟大头还是很要好的时候.虽然写的骂人家笨的文章,但是还是乐的屁颠屁颠的让她赶快打开电脑看.那时候好傻 ...

  7. Golang——延迟调用defer

    defer用于向当前函数注册稍后执行的函数调用.这些调用被称作延迟调用,它们直到当前函数执行结束前才被执行,常用于资源释放.错误处理等操作 func main() {f, err := os.Open ...

  8. FPGA学习笔记---Verilog延迟语句分析比较

    在Verilog语言中经常要用到延时语句,延时语句添加的位置不同,输出的结果就会不同.今天就来分析比较一下延时语句在不同位置时,对赋值语句的影响. 一.阻塞式左延时赋值 文件代码: `timescal ...

  9. 还在乱看 Kotlin 指南吗

    我本不太想写这篇文章,因为并没什么知识要讲,但网上乱教的实在是太多了,包括很多所谓的大牛. 虽然官网写得仍有瑕疵,但比绝大多数教程要好的多.特此给个链接, 默认读者会科学上网. Kotlin Docs ...

  10. [mysql] select查询语句大全指南

    有关mysql其他的命令语句可点击此处获取 本博客使用如下表结构作为例子讲解 create table commoditytype(ct_id int primary key auto_increme ...

最新文章

  1. NLP数据分词小整理
  2. wpf richtextbox 存储到数据库并显示
  3. Windows中如何正确认识和安装驱动程序
  4. c#如何操作excel文件、Interior.ColorIndex 色彩列表
  5. python selenium 自动登录_windows7 python3.63使用selenium+webdriver 实现自动登录使用过程...
  6. [html] html标签的属性值是否可以省略引号?为什么?
  7. springboot指定属性返回_SpringBoot中必须掌握的45个注解
  8. 字符串字母大小写转换
  9. Python使用ZeroMQ/inproc模式实现多线程服务端
  10. jk-fourm.php,JK车头灯与我的新年期望(已更新完毕)
  11. Halcon 深度学习(一):分类
  12. 广告化开发(基础知识)~广告效果指标CTR/CVR/ROI/ARPU的理解
  13. MFC界面库BCGControlBar v32.0 - 网格、报表控件升级
  14. 采用面向接口编程思想组装一台计算机
  15. 服务器,Linux,centos7成功安装显卡驱动(超详细)
  16. 百度翻译API错误码大全(建议收藏)
  17. c++实验3——个人税收计算器
  18. PHP的GD库函数大全
  19. 第二篇 在Arduino IED环境下测试ESP8266模块与外网通信
  20. 兼容iOS10资料整理

热门文章

  1. AndroidStudio3.0中butterknife报错- butterknife-7.0.1.jar (com.jakewharton:butterknife:7.0.1) Alternat
  2. 一篇学会Swagger2(集成knife4j)
  3. 2006百度之星程序设计大赛试题-变态比赛规则(解答)
  4. Java第十二天上课笔记
  5. 记一次 Showing Recent Errors Only Command /bin/sh failed with exit code 1 问题
  6. 首尔半导体美国专利诉讼胜诉,对侵犯19个LED核心技术专利的流通企业追加提起诉讼
  7. perl 报错 Can’t locate JSON.pm in @INC
  8. 国家人工智能战略的路线图
  9. 当android studio 进行Build或sync失败的时候 有一些问题其实不是问题的
  10. Vue应用openLayers实现图层数据切换显示,图层加载