对于go闭包的一点理解

一 关于匿名函数和闭包的概念,就不解释了。闭包只有引用了外部变量才成为闭包,否则就是匿名函数。

二 先看看这段代码

package main

import (
    "fmt"
    "os"
    "os/signal"
    "syscall"
)

func main() {
    for i := 0; i < 5; i++ {
        j := i
        fmt.Printf("main(),i:%d,&i:%v\r\n", i, &i)
        fmt.Printf("main(),j:%d,&j:%v\r\n", j, &j)
        go func() {
            fmt.Printf("go,i:%d,&i:%v\r\n", i, &i)
            fmt.Printf("go,j:%d,&j:%v\r\n", j, &j)
        }()
    }

    c := make(chan os.Signal, 1)
    signal.Notify(c, syscall.SIGINT, syscall.SIGTERM)
    sig := <-c
    fmt.Println(sig)
}

然后看看输出

main(),i:0,&i:0xc00001c098
main(),j:0,&j:0xc00001c0a0
main(),i:1,&i:0xc00001c098
main(),j:1,&j:0xc00001c0a8
main(),i:2,&i:0xc00001c098
main(),j:2,&j:0xc00001c0d0
main(),i:3,&i:0xc00001c098
main(),j:3,&j:0xc00001c0d8
main(),i:4,&i:0xc00001c098
main(),j:4,&j:0xc00001c0e0
go,i:5,&i:0xc00001c098
go,j:2,&j:0xc00001c0d0
go,i:5,&i:0xc00001c098
go,j:1,&j:0xc00001c0a8
go,i:5,&i:0xc00001c098
go,j:3,&j:0xc00001c0d8
go,i:5,&i:0xc00001c098
go,j:4,&j:0xc00001c0e0
go,i:2,&i:0xc00001c098
go,j:0,&j:0xc00001c0a0

三 理解

我觉得理解闭包最好的方法就是直接输出变量的地址,查看他们是否操作同一个变量地址
i 在整个进程生命期,都只有一个,无论是在闭包里面还是外面。
j 是每次for循环的时候定义了一个,所以,其实,这里是定义了 5个j的变量[可以理解为匿名变量],他们的地址各不相同。

对于闭包的理解,可以分两步分解,第一步是闭包类型的定义,第二步是闭包的调用。

在代码中,其实是有5个闭包类型的定义,5次闭包的调用。形式如下:

type Closure1 func() {
            fmt.Printf("go,i:%d,&i:%v\r\n", i, &i)
            fmt.Printf("go,j:%d,&j:%v\r\n", j, &j)
        }
Closure1()

type Closure2 func() {
            fmt.Printf("go,i:%d,&i:%v\r\n", i, &i)
            fmt.Printf("go,j:%d,&j:%v\r\n", j, &j)
        }
Closure2()

type Closure3 func() {
            fmt.Printf("go,i:%d,&i:%v\r\n", i, &i)
            fmt.Printf("go,j:%d,&j:%v\r\n", j, &j)
        }
Closure3()

type Closure4 func() {
            fmt.Printf("go,i:%d,&i:%v\r\n", i, &i)
            fmt.Printf("go,j:%d,&j:%v\r\n", j, &j)
        }
Closure4()

type Closure5 func() {
            fmt.Printf("go,i:%d,&i:%v\r\n", i, &i)
            fmt.Printf("go,j:%d,&j:%v\r\n", j, &j)
        }
Closure5()
  • 函数类型:每一个闭包类型的定义可以理解为一个函数类型的定义。
  • 函数对象:每一次闭包的调用,就是一个函数对象的生成,调用的是函数本身。
  • 闭包类型中引用的外部变量,其实就是 函数类型的成员变量。【如果对js有概念,特别类似js中的函数对象,外部变量之前,都可以加上一个this自身对象指针】
  • 闭包中定义的内部变量,和外部无关,所以,就是函数的局部变量

比如

type Closure5 func() {
            fmt.Printf("go,i:%d,&i:%v\r\n", this.i, &this.i)
            fmt.Printf("go,j:%d,&j:%v\r\n", this.j, &this.j)
        }
Closure5()

四 实践

网络上随便找的一个例子

func Adder() func(int) int{
    var x int
    return func(d int) int{
        x+=d
        return x
    }
}

func ClosureDemo5(){
    var f = Adder()
    fmt.Printf("结果=%d\n",f(1))
    fmt.Printf("结果=%d\n",f(20))
    fmt.Printf("结果=%d\n",f(300))
}

分析理解:

类型的定义 
type Closure func(d int) int{
        this.x+=d
        return this.x
    }
}
函数对象只有一个f,f有成员变量f.x

这样的理解,并没有很精确,但个人觉得,对于理解闭包是有帮助的。

学海无涯,错误难免,如有发现,尽请指正。

--the end

©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

友情链接更多精彩内容