golang中defer的关键特性示例详解
前言
大家都知道golang的defer关键字,它可以在函数返回前执行一些操作,最常用的就是打开一个资源(例如一个文件、数据库连接等)时就用defer延迟关闭改资源,以免引起内存泄漏。本文主要给大家介绍了关于golang中defer的关键特性,分享出来供大家参考学习,下面话不多说,来一起看看详细的介绍:
一、defer的作用和执行时机
go的defer语句是用来延迟执行函数的,而且延迟发生在调用函数return之后,比如
funca()int{ deferb() return0 }
b的执行是发生在return0之后,注意defer的语法,关键字defer之后是函数的调用。
二、defer的重要用途一:清理释放资源
由于defer的延迟特性,defer常用在函数调用结束之后清理相关的资源,比如
f,_:=os.Open(filename) deferf.Close()
文件资源的释放会在函数调用结束之后借助defer自动执行,不需要时刻记住哪里的资源需要释放,打开和释放必须相对应。
用一个例子深刻诠释一下defer带来的便利和简洁。
代码的主要目的是打开一个文件,然后复制内容到另一个新的文件中,没有defer时这样写:
funcCopyFile(dstName,srcNamestring)(writtenint64,errerror){ src,err:=os.Open(srcName) iferr!=nil{ return } dst,err:=os.Create(dstName) iferr!=nil{//1 return } written,err=io.Copy(dst,src) dst.Close() src.Close() return }
代码在#1处返回之后,src文件没有执行关闭操作,可能会导致资源不能正确释放,改用defer实现:
funcCopyFile(dstName,srcNamestring)(writtenint64,errerror){ src,err:=os.Open(srcName) iferr!=nil{ return } defersrc.Close() dst,err:=os.Create(dstName) iferr!=nil{ return } deferdst.Close() returnio.Copy(dst,src) }
src和dst都能及时清理和释放,无论return在什么地方执行。
鉴于defer的这种作用,defer常用来释放数据库连接,文件打开句柄等释放资源的操作。
三、defer的重要用途二:执行recover
被defer的函数在return之后执行,这个时机点正好可以捕获函数抛出的panic,因而defer的另一个重要用途就是执行recover。
recover只有在defer中使用才更有意义,如果在其他地方使用,由于program已经调用结束而提前返回而无法有效捕捉错误。
packagemain import( "fmt" ) funcmain(){ deferfunc(){ ifok:=recover();ok!=nil{ fmt.Println("recover") } }() panic("error") }
记住defer要放在panic执行之前。
四、多个defer的执行顺序
defer的作用就是把关键字之后的函数执行压入一个栈中延迟执行,多个defer的执行顺序是后进先出LIFO:
deferfunc(){fmt.Println("1")}() deferfunc(){fmt.Println("2")}() deferfunc(){fmt.Println("3")}()
输出顺序是321。
这个特性可以对一个array实现逆序操作。
五、被deferred函数的参数在defer时确定
这是defer的特点,一个函数被defer时,它的参数在defer时进行计算确定,即使defer之后参数发生修改,对已经defer的函数没有影响,什么意思?看例子:
funca(){ i:=0 deferfmt.Println(i) i++ return }
a执行输出的是0而不是1,因为defer时,i的值是0,此时被defer的函数参数已经进行执行计算并确定了。
再看一个例子:
funccalc(indexstring,a,bint)int{ ret:=a+b fmt.Println(index,a,b,ret) returnret } funcmain(){ a:=1 b:=2 defercalc("1",a,calc("10",a,b)) a=0 return }
执行代码输出
10123 1134
defer函数的参数第三个参数在defer时就已经计算完成并确定,第二个参数a也是如此,无论之后a变量是否修改都不影响。
六、被defer的函数可以读取和修改带名称的返回值
funcc()(iint){ deferfunc(){i++}() return1 }
被defer的函数是在return之后执行,可以修改带名称的返回值,上面的函数c返回的是2。
总结
以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作能带来一定的帮助,如果有疑问大家可以留言交流,谢谢大家对毛票票的支持。
参考资料
https://blog.golang.org/defer-panic-and-recover