golang中defer的基本使用教程

米米素材网

前言

第一次看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++

答案:

为什么?

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

2.输出多少

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

答案:

为什么?

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

3.输出多少

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

答案:

为什么?

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

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

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

4.输出什么

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

答案:

为什么?
 

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

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

总结

到此这篇关于golang中defer基本使用的文章就介绍到这了,更多相关go defer使用内容请搜索米米素材网以前的文章或继续浏览下面的相关文章希望大家以后多多支持米米素材网!