29、Golang 教程 - defer

什么是 defer?

defer语句的作用:含有defer语句的函数结束之前调用另外一个函数。定义看起来很复杂,我们通过一个例子就很容易理解。

 package main

import (  
    "fmt"
)

func finished() {

    fmt.Println("Finished finding largest")
}

func largest(nums []int) {

    defer finished()    
    fmt.Println("Started finding largest")
    max := nums[0]
    for _, v := range nums {

        if v > max {

            max = v
        }
    }
    fmt.Println("Largest number in", nums, "is", max)
}

func main() {

    nums := []int{

     78, 109, 2, 563, 300}
    largest(nums)
}

上述函数的功能是找出指定切片中的最大数。largest函数有一个int型切片作为参数,功能是打印输入切片中最大的数,函数第一行包含一个defer finished()语句。这意味着,函数finished()将在函数largest结束前被调用。运行程序结果如下:

 Started finding largest
Largest number in [78 109 2 563 300] is 563
Finished finding largest

largest 函数开始执行后,会打印上面的两行输出。而就在 largest将要返回的时候,又调用了我们的延迟函数Deferred Function),打印出文本 Finished finding largest

延迟方法(Defered methods)

defer 不仅限于函数的调用,调用方法也是合法的。我们写一个小程序来测试一下。

 package main

import (  
    "fmt"
)
type person struct {

    firstName string
    lastName string
}

func (p person) fullName() {

    fmt.Printf("%s %s",p.firstName,p.lastName)
}

func main() {

    p := person {

        firstName: "John",
        lastName: "Smith",
    }
    defer p.fullName()
    fmt.Printf("Welcome ")  
}

在上面的例子中,我们在第 22 行延迟了一个方法调用。其他的代码很直观,这里不再解释。该程序输出:

 Welcome John Smith  

实参取值(Arguments Evaluation)

延迟函数的参数并非在调用的时候确定,而是当执行 defer 语句的时候,就会对延迟函数的实参进行求值。

让我们通过一个例子来理解这一点。

 package main

import (  
    "fmt"
)

func printA(a int) {

    fmt.Println("value of a in deferred function", a)
}
func main() {

    a := 5
    defer printA(a)
    a = 10
    fmt.Println("value of a before deferred function call", a)

}

在上面的程序里的第 11 行,a 的初始值为 5。在第 12 行执行 defer 语句的时候,由于 a 等于 5,因此延迟函数 printA 的实参也等于 5。接着我们在第 13 行将 a 的值修改为 10。下一行会打印出 a 的值。该程序输出:

 value of a before deferred function call 10
value of a in deferred function 5

从上面的输出,我们可以看出,在调用了 defer 语句后,虽然我们将 a 修改为 10,但调用延迟函数 printA(a)后,仍然打印的是 5

defer 栈

当一个函数有多个延迟调用时,这些defer语句的调用放入到一个栈中,按照先进后出(LIFO)的顺序执行。

下面我们编写一个小程序,使用 defer 栈,将一个字符串逆序打印:

 package main

import (
    "fmt"
)

func main() {

    name := "Naveen"
    fmt.Printf("Orignal String: %s\n", string(name))
    fmt.Printf("Reversed String: ")
    for _, v := range []rune(name) {

        defer fmt.Printf("%c", v)
    }
}

在上面程序中第 11 行,for range 循环会遍历一个字符串,并在第 12 行调用了 defer fmt.Printf("%c", v)。这些延迟调用会添加到一个栈中,按照后进先出的顺序执行,因此,该字符串会逆序打印出来。该程序会输出:

 Orignal String: Naveen  
Reversed String: neevaN  

defer 的实际应用

到目前为止,我们看到的代码示例都没有体现出 defer 的实际用途。这一节我们将学习一些defer的实际应用。

defer使用的情况:当一个函数执行与当前代码流程无关。我们通过一个使用 WaitGroup 代码的示例来理解这句话的含义。我们首先会写一个没有使用 defer 的程序,然后我们会用 defer 来修改,看到 defer 带来的好处。

 package main

import (  
    "fmt"
    "sync"
)

type rect struct {

    length int
    width  int
}

func (r rect) area(wg *sync.WaitGroup) {

    if r.length < 0 {

        fmt.Printf("rect %v's length should be greater than zero\n", r)
        wg.Done()
        return
    }
    if r.width < 0 {

        fmt.Printf("rect %v's width should be greater than zero\n", r)
        wg.Done()
        return
    }
    area := r.length * r.width
    fmt.Printf("rect %v's area %d\n", r, area)
    wg.Done()
}

func main() {

    var wg sync.WaitGroup
    r1 := rect{

     -67, 89}
    r2 := rect{

     5, -67}
    r3 := rect{

     8, 9}
    rects := []rect{

     r1, r2, r3}
    for _, v := range rects {

        wg.Add(1)
        go v.area(&wg)
    }
    wg.Wait()
    fmt.Println("All go routines finished executing")
}

在上面的程序里,我们在第 8 行创建了 rect 结构体,并在第 13 行创建了 rect 的方法 area,计算出矩形的面积。area 检查了矩形的长宽是否小于零。如果矩形的长宽小于零,它会打印出对应的提示信息,而如果大于零,它会打印出矩形的面积。

main 函数创建了 3rect 类型的变量:r1r2r3。在第 34 行,我们把这 3 个变量添加到了 rects 切片里。该切片接着使用 for range 循环遍历,把 area 方法作为一个并发的 Go 协程进行调用(第 37 行)。我们用 WaitGroup wg 来确保 main函数在其他协程执行完毕之后,才会结束执行。WaitGroup 作为参数传递给 area 方法后,在第 16 行、第 21 行和第 26 行通知 main 函数,表示现在协程已经完成所有任务。如果你仔细观察,会发现 wg.Done() 只在 area 函数返回的时候才会调用,并且与代码流程无关,因此我们可以只调用一次 defer,来有效地替换掉 wg.Done() 的多次调用。

我们来用 defer 来重写上面的代码。

在下面的代码中,我们移除了原先程序中的 3wg.Done()的调用,而是用一个单独的 defer wg.Done() 来取代它(第 14 行)。这使得我们的代码更加简洁易懂。

 package main

import (  
    "fmt"
    "sync"
)

type rect struct {

    length int
    width  int
}

func (r rect) area(wg *sync.WaitGroup) {

    defer wg.Done()
    if r.length < 0 {

        fmt.Printf("rect %v's length should be greater than zero\n", r)
        return
    }
    if r.width < 0 {

        fmt.Printf("rect %v's width should be greater than zero\n", r)
        return
    }
    area := r.length * r.width
    fmt.Printf("rect %v's area %d\n", r, area)
}

func main() {

    var wg sync.WaitGroup
    r1 := rect{

     -67, 89}
    r2 := rect{

     5, -67}
    r3 := rect{

     8, 9}
    rects := []rect{

     r1, r2, r3}
    for _, v := range rects {

        wg.Add(1)
        go v.area(&wg)
    }
    wg.Wait()
    fmt.Println("All go routines finished executing")
}

该程序输出:

 rect {

     8 9}'s area 72  
rect {-67 89}'s length should be greater than zero  
rect {

     5 -67}'s width should be greater than zero  
All go routines finished executing  

在上面的程序中,使用 defer 还有一个好处。假设我们使用 if 条件语句,又给 area 方法添加了一条返回路径。如果没有使用 defer 来调用 wg.Done(),我们必须小心并确保我们在这条新添的返回路径里调用了 wg.Done()。由于现在我们延迟调用了 wg.Done(),因此无需再为这条新的返回路径添加 wg.Done() 了。