go语言中,defer语句的作用就是不管程序是否出现异常,均在函数退出时自动执行相关代码,常用于函数结束后关闭一些资源的调用,及时的释放资源。
不过,对于defer往往有两个常见的混淆点或者说误区。
误区一: 参数值
看下面一段代码:
func TestOther(t *testing.T) { i := 5 defer func() { t.Log(i) }() i = 6 return }
运行结果为:
这是因为,函数执行完成后,执行defer时,defer获取的变量i,就为该函数开始定义的i, 为同一个变量,所以输出的是改变后的i的值。
再看下面一段代码:
func TestOther(t *testing.T) { i := 5 defer func(i int) { t.Log(i) }(i) i = 6 return }
运行结果为:
可以看到,此时的运行结果,输出的为 5,这是因为,defer语句的参数值,是在defer语句出现的时候就确定下来的,而这里defer函数出现的时候,传入参数 i 此时的值是5,所以最终输出的结果为5。
误区二: return & defer
看下面一段代码:
func deferFunc(i int) (result int) { defer func() { result++ }() return i } func TestOther(t *testing.T) { i := 1 result := deferFunc(i) t.Log(result) }
运行结果为:
这是因为,函数的返回return,并不是原子的,其分为设置返回值,以及返回返回值,而上述代码中的函数 deferFunc 包含defer, 所以其返回结果的过程为: return设置返回值result=1 => defer result++ => 函数返回result , 所以最终输出的结果为 result=2