golang的defer执行时机案例分析

复制代码
package main
import "fmt"

func calcFunc(x int, y int) int {
	return x + y
}

func main() {
	// defer语句的执行顺序是,从右到左,逆序执行
	deferDemo()

	// deferDemo1函数
	demo1 := deferDemo1()
	fmt.Println(demo1) // 0

	// deferDemo2函数
	demo2 := deferDemo2()
	fmt.Println(demo2) // 1

	// deferDemo4
	demo4 := deferDemo4()
	fmt.Println(demo4) // 1

	// deferDemo5
	demo5 := deferDemo5()
	fmt.Println(demo5) // 1

	// deferDemo6
	demo6 := deferDemo6()
	fmt.Println(demo6) // 2

	// 难点解析
	/*
		   分析:
			defer注册顺序:
			1.defer fmt.Println("AA", calcFunc(x, calcFunc(x, y)))
			2.defer fmt.Println("BB", calcFunc(x, calcFunc(x, y)))
			defer执行顺序:
			1. fmt.Println("BB", calcFunc(x, calcFunc(x, y)))
			2. fmt.Println("AA", calcFunc(x, calcFunc(x, y)))

	*/
	x := 1
	y := 2
	// 注册的时候 x = 1, y = 2 所以执行的时候x = 1, y = 2
	defer fmt.Println("AA", calcFunc(x, calcFunc(x, y))) // calcFunc(1,1+2)  // 结果是:4
	x = 10
	// 注册的时候x = 10, y = 2 所以执行的时候x = 10, y = 2
	defer fmt.Println("BB", calcFunc(x, calcFunc(x, y))) // calcFunc(10,10+2) // 结果是:22
	y = 20

}

分析:

defer语句的执行顺序是,从右到左,逆序执行

执行deferDemo()函数时,会先执行defer语句,再执行函数体,函数体执行完后,再执行defer语句

执行结果:

开启defer

结束defer

defer语句6

defer语句5

defer语句4

defer语句3

defer语句2

defer语句1

复制代码
func deferDemo() {
	fmt.Println("开启defer")
	defer func() {
		fmt.Println("defer语句1")
	}()
	defer func() {
		fmt.Println("defer语句2")
	}()
	defer func() {
		fmt.Println("defer语句3")
	}()

	defer fmt.Println("defer语句4")
	defer fmt.Println("defer语句5")
	defer fmt.Println("defer语句6")

	fmt.Println("结束defer")
}

deferDemo1执行结果是 0,延迟执行

复制代码
func deferDemo1() int {
	var a int

	defer func() {
		a++
	}()

	return a
}

deferDemo1执行结果是1:原因返回匿名返回值a等待函数体内操作完成才会执行return

复制代码
func deferDemo2() (a int) {
	// 步骤一先赋值
	//a = 0

	// 步骤二再执行defer语句
	defer func() {
		a++
	}()

	// 步骤三再执行函数体
	// 返回值a等待函数体内操作完成才会执行return
	return a
}

deferDemo3执行结果是1: 原因返回匿名返回值a等待函数体内操作完成才会执行return

复制代码
func deferDemo3() (a int) {
	defer func() {
		a++
	}()

	return a
}

deferDemo4

复制代码
func deferDemo4() int {
	x := 1

	defer func(x int) {
		x++ // 内部x和外面x不是同一个变量
	}(x)

	return x // 1
}

deferDemo5

复制代码
func deferDemo5() (x int) {
	x = 1

	defer func(x int) {
		x++ // 内部x和外面x不是同一个变量
	}(x)

	return x // 1
}

deferDemo6

复制代码
func deferDemo6() (x int) {
	x = 1

	defer func() {
		x++
	}()

	return x // 2
}
相关推荐
{⌐■_■}3 小时前
【ElasticSearch】使用docker compose,通过编写yml安装es8.15和kibana可视化界面操作,go连接es
elasticsearch·docker·golang
Tony Bai1 天前
泛型重塑 Go 错误检查:errors.As 的下一站 AsA?
开发语言·后端·golang
ALex_zry1 天前
Golang云端编程深度指南:架构本质与高阶实践
开发语言·架构·golang
ALex_zry1 天前
Go语言中的迭代器模式与安全访问实践
安全·golang·迭代器模式
君万1 天前
【go语言】字符串函数
爬虫·python·golang
好学且牛逼的马2 天前
GOLANG 接口
开发语言·golang
CTRA王大大2 天前
【golang】制作linux环境+golang的Dockerfile | 如何下载golang镜像源
linux·开发语言·docker·golang
#include>2 天前
【Golang】有关垃圾收集器的笔记
笔记·golang
우리帅杰2 天前
【golang】ORM框架操作数据库
golang
秦禹辰2 天前
本地Docker部署开源Web相册图库Piwigo与在线远程访问实战方案
开发语言·后端·golang