zoukankan      html  css  js  c++  java
  • go defer详解

    一、defer 的作用和执行时机

    go 的 defer 语句是用来延迟执行函数的,而且延迟发生在调用函数 return之后,比如

    func a() int {
      defer b()
      return 0
    }
    

    b 的执行是发生在return 0之后,注意defer的语法,关键字defer之后是函数的调用。

    二、defer 的重要用途一:清理释放资源

    由于defer的延迟特性,defer常用在函数调用结束之后清理相关的资源,比如

    f, _ := os.Open(filename)
    defer f.Close()
    

    文件资源的释放会在函数调用结束之后借助defer自动执行,不需要时刻记住哪里的资源需要释放,打开和释放必须相对应。

    用一个例子深刻诠释一下defer带来的便利和简洁。

    代码的主要目的是打开一个文件,然后复制内容到另一个新的文件中,没有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 { //1
            return
        }
    
        written, err = io.Copy(dst, src)
        dst.Close()
        src.Close()
        return
    }

    代码在#1处返回之后,src文件没有执行关闭操作,可能会导致资源不能正确释放,改用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)
    }

    srcdst都能及时清理和释放,无论return在什么地方执行。

    鉴于defer的这种作用,defer常用来释放数据库连接,文件打开句柄等释放资源的操作。

    ###defer的重要用途二:执行recover

    defer的函数在return之后执行,这个时机点正好可以捕获函数抛出的panic,因而defer的另一个重要用途就是执行recover

    recover只有在defer中使用才更有意义,如果在其他地方使用,由于program已经调用结束而提前返回而无法有效捕捉错误。

    package main
    
    import (
        "fmt"
    )
    
    func main() {
        defer func() {
            if ok := recover(); ok != nil {
                fmt.Println("recover")
            }
        }()
    
        panic("error")
    
    }

    记住defer要放在panic执行之前。

    三、多个 defer 的执行顺序

    defer 的作用就是把关键字之后的函数执行压入一个栈中延迟执行,多个defer的执行顺序是后进先出LIFO

    defer func() { fmt.Println("1") }()
    defer func() { fmt.Println("2") }()
    defer func() { fmt.Println("3") }()
    

    输出顺序是 321。

    这个特性可以对一个array实现逆序操作。

    四、被 deferred 函数的参数在 defer 时确定

    这是defer的特点,一个函数被defer时,它的参数在defer时进行计算确定,即使defer之后参数发生修改,对已经defer的函数没有影响,什么意思?看例子:

    func a() {
        i := 0
        defer fmt.Println(i)
        i++
        return
    }

    a 执行输出的是 0 而不是 1,因为defer时,i 的值是 0,此时被defer的函数参数已经进行执行计算并确定了。

    再看一个例子:

    func calc(index string, a, b int) int {
        ret := a + b
        fmt.Println(index, a, b, ret)
        return ret
    }
    
    func main() {
        a := 1
        b := 2
        defer calc("1", a, calc("10", a, b))
        a = 0
        return
    }

    执行代码输出

    10 1 2 3 
    1 1 3 4

    defer函数的参数 第三个参数在defer时就已经计算完成并确定,第二个参数 a 也是如此,无论之后 a 变量是否修改都不影响。

    五、被defer的函数可以读取和修改带名称的返回值

    func c() (i int) {
        defer func() { i++ }()
        return 1
    }

    defer的函数是在return之后执行,可以修改带名称的返回值,上面的函数 c 返回的是 2。

     

    六、defer、return的执行顺序

    https://www.cnblogs.com/l199616j/p/15500631.html

    参考:https://zhuanlan.zhihu.com/p/60005467

    作者:十八岁

    -------------------------------------------

    如果觉得这篇文章对你有小小的帮助的话,记得在右下角点个“推荐”哦,博主在此感谢!

  • 相关阅读:
    POJ 3579 Median (二分)
    POJ 2976(01分数划分+二分)
    CodeForces
    hdu3555 Bomb(数位dp)
    hdu 2476 String painter(区间dp)
    poj 2955 Brackets(区间dp)
    HDU 2665(主席树,无修改第k小)
    BNUOJ ->Borrow Classroom(LCA)
    poj 2763(在线LCA+树状数组)
    Jmeter-常用线程组设置及场景运行时间计算
  • 原文地址:https://www.cnblogs.com/l199616j/p/15500696.html
Copyright © 2011-2022 走看看