如何正确使用Golang的defer关键字 讲解延迟执行的常见陷阱

golang中,defer关键字用于在函数返回前执行指定操作,但使用时需注意三个常见陷阱。首先,defer语句的参数在声明时即求值,而非执行时,因此若希望获取变量最终值,应使用闭包延迟求值。其次,在循环中频繁使用defer可能导致性能下降和内存压力,建议避免在循环体中直接使用defer或采用匿名函数限制defer作用域。最后,defer在return之后执行,若使用匿名返回值,defer修改不会影响返回结果,此时应改用命名返回值以达到预期效果。

如何正确使用Golang的defer关键字 讲解延迟执行的常见陷阱

golang中,defer关键字是一个非常实用的特性,它允许你在函数返回之前执行某些操作,比如关闭文件、释放资源等。但如果你不注意使用方式,很容易掉进一些常见的“坑”里。

如何正确使用Golang的defer关键字 讲解延迟执行的常见陷阱

defer的基本用法和执行时机

defer的作用很简单:将一个函数调用推迟到当前函数返回之前执行。不管函数是正常返回还是发生了panic,被defer的语句都会被执行。

如何正确使用Golang的defer关键字 讲解延迟执行的常见陷阱

例如:

立即学习go语言免费学习笔记(深入)”;

func main() {     defer fmt.Println("world")     fmt.Println("hello") }

输出会是:

如何正确使用Golang的defer关键字 讲解延迟执行的常见陷阱

hello world

这里的关键点在于:defer是在函数返回前执行,而不是在当前代码块结束时执行。这意味着你可以在函数中多次使用defer,它们会按照后进先出(LIFO)的顺序执行。

常见陷阱一:参数求值时机

这是最容易让人误解的地方。defer后面的函数参数会在defer语句执行时就被求值,而不是等到函数返回时才求值。

举个例子:

func badExample() {     i := 1     defer fmt.Println(i)     i++ }

很多人以为输出是2,但实际输出是1。因为i的值在defer语句执行的时候就已经确定了。

建议:如果你想延迟执行某个表达式的结果,可以考虑用闭包:defer func() { fmt.Println(i) }()

这样就能在函数返回时获取最新的i值。

常见陷阱二:在循环中使用defer可能导致性能问题

有些人喜欢在循环里用defer来处理资源释放,比如打开多个文件然后一个个defer关闭。虽然语法上没问题,但这样做会导致:

  • 每次循环都注册一个defer函数,影响性能
  • defer函数积,可能造成内存压力

比如:

for _, file := range files {     f, _ := os.Open(file)     defer f.Close() }

这段代码看似没问题,但如果循环次数很多,defer累积的数量也会很大。

建议:

  • 在循环内部避免直接使用defer
  • 可以手动管理资源,在循环结束后统一释放
  • 或者在每次循环中使用匿名函数包裹defer:
for _, file := range files {     func() {         f, _ := os.Open(file)         defer f.Close()         // do something with f     }() }

这样每个defer只作用于当前的匿名函数,不会堆积太多defer调用。

常见陷阱三:defer与return的顺序搞不清

还有一个容易混淆的地方是:defer是在return之后、函数真正退出之前执行的。

看这个例子:

func returnAndDefer() int {     var i int     defer func() {         i++     }()     return i }

这个函数返回的是0,不是1。因为return i已经把返回值确定下来了,defer修改的是变量本身,并不会影响返回结果。

建议:

  • 如果你希望defer能影响返回值,可以用命名返回值:
func namedReturn() (result int) {     defer func() {         result++     }()     return 0 }

这时返回的就是1了。


基本上就这些。defer是个好工具,但要小心它的行为细节,特别是在参数求值、循环结构和返回值上的表现。用得好了,能让你的代码更简洁清晰;用得不当,反而埋下不少隐患。

© 版权声明
THE END
喜欢就支持一下吧
点赞13 分享