简单的 defer 也有可能写出BUG

defer 也要小心使用,带来的bug

defer 很简单,但也不免大意忘记写出BUG,下面这4种情况需要注意。

1、defer 参数是"立即求值"

go 复制代码
func main() {
    i := 1
    defer fmt.Println(i) // 函数运行最后输出i的值
    i = 2
}

输出1, 原因是defer方法的参数是实时计算

2、方法接收者拷贝

go 复制代码
type Car struct { 
  name string 
}

func (c Car) Print() { 
  fmt.Println(c.name) 
}

func main() {
    c := Car{name: "Mary"}
    defer c.Print() // 最后再调用Print输出
    c.name = "Tom"
}

输出的并不是Tom,因为即使调用方法是接收器,但它前面的值也会被立即计算,所有defer处就拷贝了c,与第1点原理一样。

3、误认为是作用域名结束执行

go 复制代码
for i := 0; i < 5000; i++ {
    f, _ := os.Open(fmt.Sprintf("file_%d.csv", i))
    defer f.Close()
}

所有 Close() 都在函数结束才执行,可能造成资源浪费或文件句柄可能被耗尽。

4、未注意defer执行顺序

go 复制代码
defer fmt.Println("A")
defer fmt.Println("B")
defer fmt.Println("C")

输出的是C、B、A,defer是一个栈,先入后出。

相关推荐
日火12 小时前
Go:实现基于mutex的环形缓冲区
go
审判长烧鸡2 天前
GO错误处理【7】层层递进,环环相扣
go·报错处理
审判长烧鸡2 天前
Go结构体与指针【3】自动解引用
go·指针·结构体·自动解引用
审判长烧鸡2 天前
【GO VS PHP】之 指针/引用传递
go·php·指针·引用传递
审判长烧鸡2 天前
GO错误处理【4】报错即链条
go·异常处理·错误处理
审判长烧鸡2 天前
GO时区【1】定义与使用
go·时区
审判长烧鸡2 天前
GO错误处理【5】显式错误处理
go·错误处理·报错链条
jeff聊企业数字化2 天前
私有化即时通讯选型指南:兼顾安全与高效
go·业界资讯·即时通讯
审判长烧鸡2 天前
GO错误处理【6】显式哲学
go·显式哲学
审判长烧鸡2 天前
GO错误处理【3】返回err与日志的结合
go·架构设计·报错处理