美文网首页
golang 中的闭包和defer

golang 中的闭包和defer

作者: myonlyzzy | 来源:发表于2018-09-30 15:25 被阅读0次

golang中的defer和闭包对很多初学者来说,有时候有很多坑,但是很多介绍的文章有写的乱七八糟.放假了没事可干,就稍微总结一下.

闭包

闭包有叫匿名函数,使用闭包可以使我们的代码更加优雅简洁,顾名思义匿名函数就是没有名字的函数.

func test() int {
    var i int
    f:=func() {
        time.Sleep(1*time.Second)
        i += 1 //闭包是通过指针传递的,所以对i的修改相当于修改指针指向的内容.
        fmt.Println(" i:", i)
    }
    f()
    f()
    fmt.Println(i)
    return i
}

注意上面的代码,上面的代码输出

 i: 1
 i: 2
2
2

因为闭包中对a中的变量i是通过指针传递的,所以闭包里面对i的修改会直接修改i的值.

defer

func a() int {
    var i int
    defer add(i) //这里虽然defer是在return之前执行,但是在定义的时候,
    // 已经将defer要执行的函数压入栈,所以传递给add的是var i int的i值.
        /*defer func(){
        add(i)
    }() */
    i += 100
    return i //return  0
}
func b() int {
    var i int
    defer func(){
        add(i)
    }()
    i += 100
    return i //return  0
}
 add(i int)  {
    i += 1
    fmt.Println(i)
    
}

分别考虑一下a b 函数的输出

1 
100

这里会产生不同是因为,如果你在定义defer的时候,就要将defer后面的函数参数等入栈,等到ruturn之前的时候出栈执行,a中是将i的拷贝直接入栈,b中通过一个闭包调用,实际上将i的指针传递给闭包,闭包读取值拷贝给add.
另外我们经常说defer定义的顺序跟执行的顺序相反也是因为栈中先入后出的原因.

返回值

上面主要说了defer 和闭包,defer 容易让人混淆的地方其实是在处理返回值的时候.

func d() (i int) {
    i = 100
    defer func() {
        i += 1
    }()
    return 5 // 这里返回之前相当于 i=5 执行defer return i,所以defer中通过闭包给i++ 相当于{i=5 ,i++,return i}
}

func b() int {
    var i int
    defer func() {
        i += 1
    
    }()
    i += 100

    return i //这里相当于a=i defer return a 所以defer中闭包i++ 相当于{a=i i++ return a}
}

考虑一下上面2个函数的返回值,

100
6

b函数中返回retrun i 这个语句相当于如下

  • a=i(这里i值为100) //因为没有定义返回值名,会定义一个int 的值保存i返回.
  • defer定义的函数出栈,然后执行.在b中通过闭包对i进行了+1
  • return a
    所以返回值是100而不是101,因为在第一步a的值是100
    d 函数中return 5 这个语句相当于如下
  • i=5 //因为d函数中定义了返回值i
  • defer定义的函数出栈,然后执行.在b中通过闭包对i进行了+1
  • return i
    所以返回值是6而不是5.

总结

其实理解defer 闭包这些只要理解以下几点就会彻底明白

  1. 闭包是通过指针操作母函数中的变量
  2. defer 定义的过程就相当生成一个函数体然后将它压入栈,等到return的时候出栈执行
  3. defer 执行是在return 之前执行。这里要注意的是如果没有声明返回变量,需要先声明一个返回值然后赋值给它返回。

相关文章

  • golang 中的闭包和defer

    golang中的defer和闭包对很多初学者来说,有时候有很多坑,但是很多介绍的文章有写的乱七八糟.放假了没事可干...

  • Golang之Defer

    引用 golang defer实现原理 Golang之轻松化解defer的温柔陷阱 Golang中defer、re...

  • defer 和闭包

    今天看了一篇 文章[https://mp.weixin.qq.com/s?__biz=MzI2MDA1MTcxMg...

  • 2017-12-04

    Golang,Panic,Defer,Recover 在golang中,recover在defer里发挥作用。 一...

  • go defer 的使用和陷阱

    前言 初学 go 的同学都应该了解 defer, defer 让人又爱又恨;当 defer 和 闭包结合在一起的时...

  • 【golang】底层易错点总结

    闭包特殊性 闭包对捕获的外部变量并不是传值方式访问,而是以引用的方式访问。如以下defer 因为defer后面是一...

  • Golang中defer、闭包以及命名返回值

    defer 后面跟普通函数 defer、return 和未命名的返回值 defer、return 和命名的返回值 ...

  • Closure in Golang

    序言 Golang遵循“少即是多”的设计哲学,同时又支持闭包(Closure),那么闭包对于Golang来说肯定有...

  • go 基本函数

    多返回值函数 递归 函数类型 type 回调函数 和多态(强大) 匿名函数 与 闭包 关键字defer(defer...

  • golang

    golang中闭包的问题 熟悉go语言的开发者都知道,go的闭包是可以记录状态的。当我想用闭包来实现保存某一str...

网友评论

      本文标题:golang 中的闭包和defer

      本文链接:https://www.haomeiwen.com/subject/yqswoftx.html