defer 语句的用途是:含有 defer 语句的函数,会在该函数将要返回之前,调用另一个函数。这个定义可能看起来很复杂,我们通过一个示例就很容易明白了。
函数中使用defer
代码语言:javascript复制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)
}
不仅函数中能使用defer 方法中也能使用defer
代码语言:javascript复制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 ")
}
实参取值使用defer
代码语言:javascript复制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)
}
image.png
在上面的程序里的,a 的初始值为 5,执行 defer 语句的时候,由于 a 等于 5,因此延迟函数 printA 的实参也等于 5。接着将 a 的值修改为 10。下一行会打印出 a 的值。
defer 栈的使用
当一个函数内多次调用 defer 时,Go 会把 defer 调用放入到一个栈中,随后按照后进先出(Last In First Out, LIFO)的顺序执行。
我们下面编写一个小程序,使用 defer 栈,将一个字符串逆序打印。
代码语言:javascript复制package main
import (
"fmt"
)
func main() {
name := "Naveen"
fmt.Printf("Orignal String: %sn", string(name))
fmt.Printf("Reversed String: ")
for _, v := range []rune(name) {
defer fmt.Printf("%c", v)
}
}
defer 的实际应用
代码语言:javascript复制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 zeron", r)
wg.Done()
return
}
if r.width < 0 {
fmt.Printf("rect %v's width should be greater than zeron", r)
wg.Done()
return
}
area := r.length * r.width
fmt.Printf("rect %v's area %dn", 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")
}
注意 wg.Done() 我们在area 中写了三次,我们能不能只写一次了
优化后如下
代码语言:javascript复制func (r rect) area(wg *sync.WaitGroup) {
defer wg.Done()
if r.length < 0 {
fmt.Printf("rect %v's length should be greater than zeron", r)
return
}
if r.width < 0 {
fmt.Printf("rect %v's width should be greater than zeron", r)
return
}
area := r.length * r.width
fmt.Printf("rect %v's area %dn", r, area)
}