作者简介:
高科,先后在 IBM PlatformComputing从事网格计算,淘米网,网易从事游戏服务器开发,拥有丰富的C++,go等语言开发经验,mysql,mongo,redis等数据库,设计模式和网络库开发经验,对战棋类,回合制,moba类页游,手游有丰富的架构设计和开发经验。 (谢谢你的关注)-------------------------------------------------------------------------------------------------------------------------------
从名字就可以推断出,Go编程语言中使用的defer关键字是用来延迟执行一个要执行的函数(方法)的。
了解 Defer 及其目的
在许多编程语言中,defer
类似于ensure
或 之类的概念finally
。它允许设置一个函数调用,无论当前函数如何退出,无论是由于成功执行还是panic,该函数调用都将被执行。主要目的defer
是提高代码的可读性、可维护性和资源管理。通过defer清理任务,你可以保持代码井井有条,并确保有效处理资源密集型操作。
defer的使用
使用的语法defer
很简单:
defer functionCall(arguments)
defer
每当调用包含语句的函数时,指定的语句functionCall
不会立即执行。相反,它被添加到延迟函数列表中,一旦封闭函数返回,这些函数将以相反的顺序执行。
首先我们来看下这段代码:
package main
import "fmt"
func strWorld() string {
return "World!"
}
func strHello() string {
return "Hello"
}
func main() {
fmt.Println("Let's check defer keyword")
defer fmt.Println(strWorld())
fmt.Println(strHello())
}
结果如下:
Let's check defer keyword
Hello
World!
defer 具有类似堆栈的 LIFO(List In Fist Out)结构,因此后面添加的 defer 会先执行。
如果将示例代码更改如下:
package main
import "fmt"
func strWorld() string {
return "World!"
}
func strWorld2() string {
return "World 2"
}
func strHello() string {
return "Hello"
}
func main() {
fmt.Println("Let's check defer keyword")
defer fmt.Println(strWorld())
fmt.Println(strHello())
defer fmt.Println(strWorld2())
}
结果变化如下所示。
Let's check defer keyword
Hello
World 2
World!
可以看到先执行strWorld2(),后执行strWorld()。
因panic而推迟
defer
也可以在优雅地处理panic方面发挥至关重要的作用。当发生panic时,defer函数在程序终止之前仍然会执行。可以利用它来执行必要的清理任务或在从panic中恢复时记录有用的信息。下面是一个代码片段来演示这一点:
package main
import "fmt"
func main() {
defer func() {
if r := recover(); r != nil {
fmt.Println("Recovered from panic:", r)
}
}()
panic("Something went terribly wrong!")
}
在此示例中,延迟函数捕获紧急消息并在程序panic之前打印它,为调试提供有价值的信息。
defer方法调用
该defer
语句不仅适用于函数,也适用于方法调用。你可以推迟方法的执行,就像使用常规函数一样。当你想要在方法执行结束时执行操作(无论方法的结果如何)时,这非常有用。让我们看一个片段来演示这一点:
package main
import "fmt"
type Counter struct {
count int
}
func (c *Counter) Increment() {
c.count++
defer c.printCount()
}
func (c *Counter) printCount() {
fmt.Println( "当前计数:" , c.count)
}
func main () {
counter := &Counter{}
counter.Increment()
}
在此示例中,该printCount
方法在方法内被延迟Increment
。因此,计数将在Increment
方法执行结束时打印。
defer特性:
根据刚才的代码示例,我们对defer的特性做一个简单的归纳总结:
1. 关键字 defer 用于注册延迟调用。
2. 这些调用直到 return 前才被执。因此,可以用来做资源清理。
3. 多个defer语句,按先进后出的方式执行。
4. defer语句中的变量,在defer声明时就决定了。
defer用途:
1. 关闭文件句柄
2. 锁资源释放
3. 数据库连接释放
4. 一些异常捕获
让我们深入研究一个代码示例,演示如何defer
在 Go 中进行文件句柄管理。
package main
import (
"fmt"
"os"
)
func main () {
f := createFile( "/tmp/defer.txt" )
defer closeFile(f)
writeFile(f)
}
func createFile (p string ) *os.File {
fmt.Println( "创建" )
f, err := os.Create(p)
if err != nil {
panic (err)
}
return f
}
func writeFile (f *os.File) {
fmt.Println( "写入" )
fmt.Fprintln(f, "data" )
}
func closeFile (f *os.File) {
fmt.Println( "关闭" )
err := f.Close()
if err != nil {
fmt.Fprintf(os .Stderr, "错误: %v\n" , err)
os.Exit( 1 )
}
}
当运行此代码时,你将看到以下输出:
创建
写入
结束
Defer 如何简化资源管理
在给定的示例中,我们创建一个文件,向其中写入数据,并在写入完成后关闭该文件。我们来分解一下这个过程:
- 我们调用该
createFile
函数,它返回一个文件对象并将“creating”打印到控制台。 - 在退出之前
createFile
,我们推迟 的执行closeFile(f)
。这确保了当所有其他语句都已执行时,该closeFile
函数将在封闭函数 ( ) 的末尾被调用。main
- 接下来,我们调用该
writeFile
函数,它将“数据”写入文件并将“写入”打印到控制台。 - 最后,当
main
即将返回时,closeFile(f)
执行 defer,在控制台打印“close”,并有效地关闭文件。
使用 Defer 处理错误
关闭文件或释放其他资源时,正确处理错误至关重要。即使在延迟函数中,也必须检查错误以防止任何意外行为。在我们的示例中,该closeFile
函数在使用 关闭文件时检查错误f.Close()
。如果发生错误,则会显示该错误,并且程序将退出并显示错误代码。
此错误处理可确保程序运行可靠,并提供足够的信息来诊断和修复清理期间发生的问题。
defer的陷阱
defer的功能很强大,对于资源管理非常方便,但是如果没用好,也会有陷阱。
defer 碰上闭包
因此如果先前面的资源先释放了,后面的语句就没法执行了。
package main
import "fmt"
func main() {
var whatever [5]struct{}
for i := range whatever {
defer fmt.Println(i)
}
}
输出结果:
4
3
2
1
0
我们改成闭包,看看
package main
import "fmt"
func main() {
var whatever [5]struct{}
for i := range whatever {
defer func() { fmt.Println(i) }()
}
}
输出结果:
4
4
4
4
4
其实go说的很清楚,函数正常执行,由于闭包用到的变量 i 在执行的时候已经变成4,所以输出全都是4.
defer f.Close
这个大家用的都很频繁,但是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
这个输出并不会像我们预计的输出c b a,而是输出c c c
可是按照前面的go spec中的说明,应该输出c b a才对啊.
那我们换一种方式来调用一下.
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
这个时候输出的就是c b a
当然,如果你不想多写一个函数,也很简单,可以像下面这样,同样会输出c b a
看似多此一举的声明
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 {
t2 := t
defer t2.Close()
}
}
输出结果:
c closed
b closed
a closed
通过以上例子,可以得出下面的结论:
使用 defer 的函数的参数是在执行 defer 语句时计算的,而不是在调用函数时计算的。defer后面的语句在执行的时候,函数调用的参数会被保存起来,但是不执行。也就是复制了一份。但是并没有说struct这里的this指针如何处理,通过这个例子可以看出go语言并没有把这个明确写出来的this指针当作参数来看待。
多个 defer 注册,按 FILO 次序执行 ( 先进后出 )。哪怕函数或某个延迟调用发生错误,这些调用依旧会被执行。
package main
func test(x int) {
defer println("a")
defer println("b")
defer func() {
println(100 / x) // div0 异常未被捕获,逐步往外传递,最终终止进程。
}()
defer println("c")
}
func main() {
test(0)
}
输出结果:
c
b
a
panic: runtime error: integer divide by zero
*
延迟调用参数在注册时求值或复制,可用指针或闭包 “延迟” 读取。
package main
func test() {
x, y := 10, 20
defer func(i int) {
println("defer:", i, y) // y 闭包引用
}(x) // x 被复制
x += 10
y += 100
println("x =", x, "y =", y)
}
func main() {
test()
}
输出结果:
x = 20 y = 120
defer: 10 120
*
滥用 defer 可能会导致性能问题,尤其是在一个 “大循环” 里。
package main
import (
"fmt"
"sync"
"time"
)
var lock sync.Mutex
func test() {
lock.Lock()
lock.Unlock()
}
func testdefer() {
lock.Lock()
defer lock.Unlock()
}
func main() {
func() {
t1 := time.Now()
for i := 0; i < 10000; i++ {
test()
}
elapsed := time.Since(t1)
fmt.Println("test elapsed: ", elapsed)
}()
func() {
t1 := time.Now()
for i := 0; i < 10000; i++ {
testdefer()
}
elapsed := time.Since(t1)
fmt.Println("testdefer elapsed: ", elapsed)
}()
}
输出结果:
test elapsed: 223.162µs
testdefer elapsed: 781.304µs
defer 与 文件关闭
让我们演示一个文件复制的例子:函数需要打开两个文件,然后将其中一个文件的内容复制到另一个文件:
func CopyFile(dstName, srcName string) (written int64, err error) {
src, err := os.Open(srcName)
if err != nil {
return
}
dst, err := os.Create(dstName)
if err != nil {
return
}
written, err = io.Copy(dst, src)
dst.Close()
src.Close()
return
}
上面的代码虽然能够工作,但是隐藏一个bug。如果第一个os.Open
调用成功,但是第二个os.Create
调用失败,那么会在没有释放src
文件资源的情况下返回。虽然我们可以通过在第二个返回语句前添加src.Close()
调用来修复这个BUG;但是当代码变得复杂时,类似的问题将很难被发现和修复。我们可以通过defer
语句来确保每个被正常打开的文件都能被正常关闭:
func CopyFile(dstName, srcName string) (written int64, err error) {
src, err := os.Open(srcName)
if err != nil {
return
}
defer src.Close()
dst, err := os.Create(dstName)
if err != nil {
return
}
defer dst.Close()
return io.Copy(dst, src)
}
defer
语句可以让我们在打开文件时马上思考如何关闭文件。不管函数如何返回,文件关闭语句始终会被执行。同时defer
语句可以保证,即使io.Copy
发生了异常,文件依然可以安全地关闭。
defer 与 return
package main
import "fmt"
func foo() (i int) {
i = 0
defer func() {
fmt.Println(i)
}()
return 2
}
func main() {
foo()
}
输出结果:
2
解释:在有具名返回值的函数中(这里具名返回值为 i),执行 return 2 的时候实际上已经将 i 的值重新赋值为 2。所以defer closure 输出结果为 2 而不是 1。
defer nil 函数
package main
import (
"fmt"
)
func test() {
var run func() = nil
defer run()
fmt.Println("runs")
}
func main() {
defer func() {
if err := recover(); err != nil {
fmt.Println(err)
}
}()
test()
}
输出结果:
runs
runtime error: invalid memory address or nil pointer dereference
解释:名为 test 的函数一直运行至结束,然后 defer 函数会被执行且会因为值为 nil 而产生 panic 异常。然而值得注意的是,run() 的声明是没有问题,因为在test函数运行完成后它才会被调用。
在错误的位置使用 defer
当 http.Get 失败时会抛出异常。
package main
import "net/http"
func do() error {
res, err := http.Get("http://www.google.com")
defer res.Body.Close()
if err != nil {
return err
}
// ..code...
return nil
}
func main() {
do()
}
输出结果:
panic: runtime error: invalid memory address or nil pointer dereference
因为在这里我们并没有检查我们的请求是否成功执行,当它失败的时候,我们访问了 Body 中的空变量 res ,因此会抛出异常
解决方案
总是在一次成功的资源分配下面使用 defer ,对于这种情况来说意味着:当且仅当 http.Get 成功执行时才使用 defer
package main
import "net/http"
func do() error {
res, err := http.Get("http://xxxxxxxxxx")
if res != nil {
defer res.Body.Close()
}
if err != nil {
return err
}
// ..code...
return nil
}
func main() {
do()
}
在上述的代码中,当有错误的时候,err 会被返回,否则当整个函数返回的时候,会关闭 res.Body 。
解释:在这里,你同样需要检查 res 的值是否为 nil ,这是 http.Get 中的一个警告。通常情况下,出错的时候,返回的内容应为空并且错误会被返回,可当你获得的是一个重定向 error 时, res 的值并不会为 nil ,但其又会将错误返回。上面的代码保证了无论如何 Body 都会被关闭,如果你没有打算使用其中的数据,那么你还需要丢弃已经接收的数据。
不检查错误
在这里,f.Close() 可能会返回一个错误,可这个错误会被我们忽略掉
package main
import "os"
func do() error {
f, err := os.Open("book.txt")
if err != nil {
return err
}
if f != nil {
defer f.Close()
}
// ..code...
return nil
}
func main() {
do()
}
改进一下
package main
import "os"
func do() error {
f, err := os.Open("book.txt")
if err != nil {
return err
}
if f != nil {
defer func() {
if err := f.Close(); err != nil {
// log etc
}
}()
}
// ..code...
return nil
}
func main() {
do()
}
再改进一下
通过命名的返回变量来返回 defer 内的错误。
package main
import "os"
func do() (err error) {
f, err := os.Open("book.txt")
if err != nil {
return err
}
if f != nil {
defer func() {
if ferr := f.Close(); ferr != nil {
err = ferr
}
}()
}
// ..code...
return nil
}
func main() {
do()
}
释放相同的资源
如果你尝试使用相同的变量释放不同的资源,那么这个操作可能无法正常执行。
package main
import (
"fmt"
"os"
)
func do() error {
f, err := os.Open("book.txt")
if err != nil {
return err
}
if f != nil {
defer func() {
if err := f.Close(); err != nil {
fmt.Printf("defer close book.txt err %v\n", err)
}
}()
}
// ..code...
f, err = os.Open("another-book.txt")
if err != nil {
return err
}
if f != nil {
defer func() {
if err := f.Close(); err != nil {
fmt.Printf("defer close another-book.txt err %v\n", err)
}
}()
}
return nil
}
func main() {
do()
}
输出结果:
defer close book.txt err close ./another-book.txt: file already closed
当延迟函数执行时,只有最后一个变量会被用到,因此,f 变量 会成为最后那个资源 (another-book.txt)。而且两个 defer 都会将这个资源作为最后的资源来关闭
解决方案:
package main
import (
"fmt"
"io"
"os"
)
func do() error {
f, err := os.Open("book.txt")
if err != nil {
return err
}
if f != nil {
defer func(f io.Closer) {
if err := f.Close(); err != nil {
fmt.Printf("defer close book.txt err %v\n", err)
}
}(f)
}
// ..code...
f, err = os.Open("another-book.txt")
if err != nil {
return err
}
if f != nil {
defer func(f io.Closer) {
if err := f.Close(); err != nil {
fmt.Printf("defer close another-book.txt err %v\n", err)
}
}(f)
}
return nil
}
func main() {
do()
}