golang中defer的基本使用教程

来自:网络
时间:2023-01-01
阅读:
目录

前言

第一次看go基础语法的时候,用使用到了defer。但是一直不知道它到底是什么,有什么用途。这几天通过查询、学习。算是对defer有了一点浅显的认识。

1.什么是defer

defer是go中一种延迟调用机制,defer后面的函数只有在当前函数执行完毕后才能执行,通常用于释放资源。

2.defer的特点

defer遵循先进后出的原则,类似于栈的结构。

补充下:为什么要把defer设计成这种机制?

因为后申请的资源和可能对前面申请的资源有依赖。如果先将前面申请的资源释放掉了。对于后面的资源可能会造成影响。所以先释放后申请的资源,再释放前面申请的资源。

3.defer什么时间执行

前面说到,defer只有在当前函数执行完毕后,才会执行。其实不太准确。

go中的return语句并不是原子性操作,一般是分为两步:

  • 将返回值赋值给一个变量
  • 执行RET指令

defer就执行在1之后,2之前。

4.defer常见的坑

1.输出是多少?

	x := 10
	defer func(a int) {
		fmt.Println(a)
	}(x)
	x++

答案:

golang中defer的基本使用教程

为什么?

因为defer后面的函数在入栈的时候保存的是入栈那一刻的值,而当时x的值是10,所以后期对x修改,并不会影响栈内函数的值。

2.输出多少

	x := 10
	defer func(a *int) {
		fmt.Println(*a)
	}(&x)
	x++

答案:

golang中defer的基本使用教程

为什么?

这里defer后面函数入栈的时候存入的执行变量x的指针。所以,后期x值改变的时候,输出结果也会改变。

3.输出多少

func test()(x int)  {
	 x = 10
	 defer func() {
	 	x++
	 }()
	 return x
}

答案:

golang中defer的基本使用教程

为什么?

之前我们说过,return并不是原子性操作,是通过一个变量赋值和ret指令来完成的。

而上述例子中,是具名函数。即返回值带有名字。这样我们在执行defer的时候相当于修改了返回值的值。所以为11

看到这里,博主想到了闭包。和闭包有没有关系呢?

4.输出什么

func test1() int {
	x := 10
	defer func() {
		x++
	}()
	// ans = x
	// -------- defer x = x+1
	// return x
	return x
}

答案:

golang中defer的基本使用教程

为什么?
 

还是return语句的原因,博主已经在代码中给出提示。可见,非具名函数不会受到相应的影响。

对于defer暂时理解了这些,下次再见。

总结

返回顶部
顶部