0
Golang有一个特殊的控件语句,那就是defer,defer语句用于延迟调用指定的函数,比如释放资源等,它会在函数最后执行,但在return之前,先让我们看代码简单了解下:
package main func main() { test() }func test() { println("test1") defer func() { println("defer test2") }() println("test3") }
执行结果如下:
test1 test3 defer test2
很清楚地看到,有defer那个函数最后才执行的
现在改一下代码,让代码panic,当抛出异常时,defer延迟函数还会执行吗?
package main func main() { test() }func test() { println("test1") panic("panic") defer func() { println("defer test2") }() println("test3") }
输出
test1 panic: panic Process finished with exit code 2
延迟函数居然没有执行,为什么会这样呢?是因为panic在延迟函数之前了,再改一下代码如下
package main func main() { test() }func test() { println("test1") defer func() { println("defer test2") }() panic("panic") println("test3") }
输出
test1 defer test2 panic: panic
延迟函数执行了,可以看到panic在延迟函数之前延迟函数是执行不了的,毕竟抛出异常。
1
再看一个例子
package main func main() { test() }func test() { for i:=0;i<5 ;i++ { defer func() { println(i) }() } }
输出
5 5 5 5 5 Process finished with exit code 0
是不是觉得很惊诧啊,是的,为什么不是0 1 2 3 4 啊?
这正是因为延迟函数执行时机引起的,当println(i)时,i已经是5了,跳出了for循环,如果延迟函数要使用外部的变量应该通过参数传入,看下面
package main func main() { test() }func test() { for i := 0; i < 5; i++ { defer func(i int) { println(i) }(i) } }
输出
4 3 2 1 0 Process finished with exit code 0
对了吧,哎哎哎,不对啊,怎么反过来了?
其实延迟函数是一个栈,先进后出最后放进去的最后出来,我们再试一个是不是这样
package main func main() { test() }func test() { defer func() { println("test1") }() defer func() { println("test2") }() }
输出
test2 test1 Process finished with exit code 0
果真如此。
2.总结
看了上边的例子,最后我们总结一下,学过的东西如果不好好总结,会很快忘记的,打算以后看的书都全部写出来,这样书才没有白读啊
我们规定一下,上边test()方法为外围函数,调用test()的叫调用函数,调用与被调用区分开,总结如下
当外围函数正常执行完毕时,只有其中的延迟函数都执行完毕时,外围函数才会真正执行完
当外围函数执行return时,只有延迟函数全都执行完时,才会真正地返回
panic在延迟函数后边,只有延迟函数执行完毕时,panci才会扩散到调用函数
延迟函数总是会在外围函数执行前执行,除非延迟函数前面已经panic了
defer语句在外围函数的函数体中位置不限,数量不限
如果延迟函数要使用外部的变量应该通过参数传入
迟延函数是一个栈,先进后出
作者:想入飞飞___
链接:https://www.jianshu.com/p/772b9a1220ae
热门评论
defer在return之后执行,不是之前