Golang defer关键字

defer特性

1.关键字defer 用于注册延迟调用

2.这些调用直到return 前才被执行。因此,可以用来做资源清理

3.多个defer语句,按先进后出的方式执行

4.defer语句中的变量,在defer生命时就决定了

defer用途

1.关闭文件句柄

2.锁资源释放

3.数据库连接释放

defer用例

1.多个defer输出顺序
Go 复制代码
func main() {
	for i := 0; i < 3; i++ {
		defer fmt.Println(i) // 1. 这三个 defer 语句会按照 i=0, i=1, i=2 的顺序被推入栈中。
	}
	fmt.Println("循环结束") // 2. 循环结束,打印 "循环结束"。

	if true {
		defer fmt.Println("条件语句中的 defer") // 3. 这个 defer 语句被推入栈中。
	}

	defer fmt.Println("最后的 defer") // 4. 这个 defer 语句被推入栈中。

	fmt.Println("main 函数执行中") // 5. 打印 "main 函数执行中"。
}

//结果
//循环结束
//main 函数执行中
//最后的 defer
//条件语句中的 defer
//2
//1
//0
2.defer遇到循环
Go 复制代码
//第一个例子
package main

import "fmt"

type Test struct {
    name string
}

func (t *Test) Close() {
    fmt.Println(t.name, " closed")
}
func main() {
    ts := []Test{{"a"}, {"b"}, {"c"}}
    for _, t := range ts {
        defer t.Close()
    }
}

//结果:c closed 
//c closed
//c closed
Go 复制代码
//第二个例子
package main

import "fmt"

type Test struct {
    name string
}

func (t *Test) Close() {
    fmt.Println(t.name, " closed")
}
func Close(t Test) {
    t.Close()
}
func main() {
    ts := []Test{{"a"}, {"b"}, {"c"}}
    for _, t := range ts {
        defer Close(t)
    }
}
//结果
//c  closed
//b  closed
//a  closed

第一个例子中,当defer语句被执行时,它都会捕获t当前值的副本,并不是t的地址。由于t是在循环遍历的,每次迭代都会创建t的副本,而不是t的地址。由于t是在循环中逐一遍历的,每次迭代都会创建t的新副本。因此当main函数结束时,defer语句会按照后进先出的顺序执行,但是每个defer调用都会打印循环结束时t的最终副本,即"c"。

第二个例子中的close函数接受一个Test类型的参数,并且调用这个参数的Close方法。由于Close是一个单独的函数,它接收t的副本作为参考。当defer调用Close(t)时,它传递了t的当前值的副本给Close函数。这意味着每个defer调用都捕获了循环中t的当前值的副本。

3.defer +recover 捕获异常
Go 复制代码
package main

import "fmt"

func safeDivide(a, b int) (int, error) {
    defer func() {
        if r := recover(); r != nil {
            fmt.Println("Recovered from panic:", r)
        }
    }()
    
    if b == 0 {
        return 0, fmt.Errorf("cannot divide by zero")
    }
    
    return a / b, nil
}

func main() {
    result, err := safeDivide(10, 0)
    if err != nil {
        fmt.Println("Error:", err)
    }
    fmt.Println("Result:", result)
}
4.defer 遇到闭包
Go 复制代码
package main

import "fmt"

func main() {
    var whatever [5]struct{}
    for i := range whatever {
        defer func() { fmt.Println(i) }()
    }
}
//结果
//4
//4
//4
//4
//4

这是因为闭包用到的变量i在执行时就已经变成4了,所以输出全是4

注意事项

注意事项

  1. 性能 :过度使用 defer 可能会影响性能,因为每个 defer 调用都会创建额外的栈帧。

  2. 控制流defer 语句的执行顺序可能与直觉相反,需要仔细设计以确保逻辑正确。

  3. 错误处理defer 语句中的错误处理代码应该能够处理所有可能的错误情况。

  4. 嵌套使用 :嵌套使用 defer 时,确保外层的 defer 能够正确处理内层可能抛出的错误。

相关推荐
yesyesido5 分钟前
智能文件格式转换器:文本/Excel与CSV无缝互转的在线工具
开发语言·python·excel
_200_7 分钟前
Lua 流程控制
开发语言·junit·lua
环黄金线HHJX.7 分钟前
拼音字母量子编程PQLAiQt架构”这一概念。结合上下文《QuantumTuan ⇆ QT:Qt》
开发语言·人工智能·qt·编辑器·量子计算
王夏奇8 分钟前
python在汽车电子行业中的应用1-基础知识概念
开发语言·python·汽车
He_Donglin8 分钟前
Python图书爬虫
开发语言·爬虫·python
qq_2562470512 分钟前
除了“温度”,如何用 Penalty (惩罚) 治好 AI 的“复读机”毛病?
后端
星融元asterfusion18 分钟前
AsterNOS SONiC基于YANG模型的现代网络管理:从CLI到gNMI的演进
开发语言·sonic·yang
web3.088899920 分钟前
1688商品详情API接口深度解析
开发语言·python
内存不泄露23 分钟前
基于Spring Boot和Vue 3的智能心理健康咨询平台设计与实现
vue.js·spring boot·后端
qq_124987075324 分钟前
基于Spring Boot的电影票网上购票系统的设计与实现(源码+论文+部署+安装)
java·大数据·spring boot·后端·spring·毕业设计·计算机毕业设计