6.4 defer 和追踪

    关键字 defer 的用法类似于面向对象编程语言 Java 和 C# 的 finally 语句块,它一般用于释放某些已分配的资源。

    示例 6.8 :

    输出:

    1. In Function1 at the top
    2. In Function1 at the bottom!
    3. Function2: Deferred until the end of the calling function!

    请将 defer 关键字去掉并对比输出结果。

    使用 defer 的语句同样可以接受参数,下面这个例子就会在执行 defer 语句时打印 0

    1. func a() {
    2. i := 0
    3. defer fmt.Println(i)
    4. i++
    5. return
    6. }

    当有多个 defer 行为被注册时,它们会以逆序执行(类似栈,即后进先出):

    1. func f() {
    2. for i := 0; i < 5; i++ {
    3. defer fmt.Printf("%d ", i)
    4. }
    5. }

    关键字 defer 允许我们进行一些函数执行完成后的收尾工作,例如:

    1. 关闭文件流 (详见 第 12.2 节
    1. // open a file
    2. defer file.Close()
    1. 解锁一个加锁的资源 (详见 )
    1. 打印最终报告
    1. printHeader()
    2. defer printFooter()
    1. 关闭数据库链接
    1. // open a database connection
    2. defer disconnectFromDB()

    合理使用 defer 语句能够使得代码更加简洁。

    以下代码模拟了上面描述的第 4 种情况:

    1. package main
    2. import "fmt"
    3. func main() {
    4. doDBOperations()
    5. }
    6. func connectToDB() {
    7. fmt.Println("ok, connected to db")
    8. }
    9. func disconnectFromDB() {
    10. fmt.Println("ok, disconnected from db")
    11. }
    12. func doDBOperations() {
    13. connectToDB()
    14. fmt.Println("Doing some DB operations ...")
    15. fmt.Println("Oops! some crash or network error ...")
    16. fmt.Println("Returning from function here!")
    17. return //terminate the program
    18. // deferred function executed here just before actually returning, even if
    19. // there is a return or abnormal termination before
    20. }

    输出:

    1. ok, connected to db
    2. Defering the database disconnect.
    3. Doing some DB operations ...
    4. Oops! some crash or network error ...
    5. Returning from function here!
    6. ok, disconnected from db

    使用 defer 语句实现代码追踪

    一个基础但十分实用的实现代码执行追踪的方案就是在进入和离开某个函数打印相关的消息,即可以提炼为下面两个函数:

    示例 6.10 defer_tracing.go:

    1. package main
    2. import "fmt"
    3. func trace(s string) { fmt.Println("entering:", s) }
    4. func untrace(s string) { fmt.Println("leaving:", s) }
    5. func a() {
    6. trace("a")
    7. defer untrace("a")
    8. fmt.Println("in a")
    9. }
    10. func b() {
    11. trace("b")
    12. defer untrace("b")
    13. fmt.Println("in b")
    14. a()
    15. }
    16. func main() {
    17. b()
    18. }

    输出:

    1. entering: b
    2. in b
    3. entering: a
    4. in a
    5. leaving: a

    上面的代码还可以修改为更加简便的版本(示例 6.11 ):

    1. package main
    2. func trace(s string) string {
    3. fmt.Println("entering:", s)
    4. return s
    5. }
    6. func un(s string) {
    7. fmt.Println("leaving:", s)
    8. }
    9. func a() {
    10. defer un(trace("a"))
    11. fmt.Println("in a")
    12. }
    13. func b() {
    14. defer un(trace("b"))
    15. fmt.Println("in b")
    16. a()
    17. }
    18. func main() {
    19. b()
    20. }

    使用 defer 语句来记录函数的参数与返回值

    下面的代码展示了另一种在调试时使用 defer 语句的手法(示例 6.12 defer_logvalues.go):

    1. package main
    2. import (
    3. "io"
    4. "log"
    5. )
    6. func func1(s string) (n int, err error) {
    7. defer func() {
    8. log.Printf("func1(%q) = %d, %v", s, n, err)
    9. }()
    10. return 7, io.EOF
    11. }
    12. func main() {
    13. }

    输出: