golang中defer的些许总结
Posted zuixime0515
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了golang中defer的些许总结相关的知识,希望对你有一定的参考价值。
引言
最近自己在学习go项目的时候,查看别人的源码经常发现defer
的使用,加上自己之前对这个关键字没怎么在意,只知道差不多是用来释放内部变量的。
查看资料可以发现,defer
会在当前函数或者方法返回之前执行传入的函数,经常用于关闭文件描述符、关闭数据库连接以及解锁资源。
为了能够更形象地理解defer
的行为,我就在这贴上一段关闭文件描述符的示例代码:
func?CopyFile(source,?dest?string)?bool?{
????if?source?==?""?||?dest?==?""?{
????????log.Println("source?or?dest?is?null")
????????return?false
????}
????source_open,?err?:=?os.Open(source)
????if?err?!=?nil?{
????????log.Println(err.Error())
????????return?false
????}
????defer?source_open.Close()
????dest_open,?err?:=?os.OpenFile(dest,?os.O_CREATE|os.O_WRONLY,?644)
????if?err?!=?nil?{
????????log.Println(err.Error())
????????return?false
????}
????defer?dest_open.Close()
????_,?copy_err?:=?io.Copy(dest_open,?source_open)
????if?copy_err?!=?nil?{
????????log.Println(copy_err.Error())
????????return?false
????}?else?{
????????return?true
????}
}
在示例代码中,使用了defer
来关闭文件流,用来避免忘记文件释放的问题。
defer的调用时机以及多次调用defer的执行顺序
defer
传入的函数会在当前调用方法返回之前得到运行。
为了更好地解释上面的原理,通过一个简单例子来观察就很明显了:
func?main()?{
????for?i?:=?0;?i?<?4;?i++?{
????????defer?fmt.Println(i)
????}???
}
示例程序中是用来循环输出数字的,实际上的运行结果如下,可以观察得出defer
的效果类似于栈的先进后出
:
3
2
1
0
defer的预计算参数
defer继承了Go中函数调用的特性,就是函数调用都是传值的。所以,使用defer关键字之后,会立刻对函数中引用的外部参数进行拷贝
示例程序如下,最后包含了输出结果:
func?PrintInfo()?{
????val?:=?0
????fmt.Println("use?defer?to?print?val-->")
????//?in?here,?will?output?0
????defer?fmt.Println(val)
????val++
????//?in?here,?will?output?1
????defer?fmt.Println(val)
????return
}
use defer to print val-->
1
0
至于程序为什么会输出1 0 而不是 0 1 ,答案已经在上面有了解释;
参考资料
- https://draveness.me/golang/docs/part2-foundation/ch05-keyword/golang-defer/#53-defer
- https://studygolang.com/articles/10167
以上是关于golang中defer的些许总结的主要内容,如果未能解决你的问题,请参考以下文章