defer func for循环,为啥结果不是012,而是3 3 3呢,麻烦大佬解答下,百度无果,小白问题,打扰大家了。

lishuailw · 2020-08-22 15:44:50 · 1473 次点击 · 大约8小时之前 开始浏览    置顶
这是一个创建于 2020-08-22 15:44:50 的主题,其中的信息可能已经有所发展或是发生改变。

func main() {
    for i := 0; i < 3; i++ {
        defer func(){
            fmt.Println(i) 
        }()
    }
}

输出结果:
3
3
3

我理解的输出结果:
2
1
0



请问为什么是输出 3 3 3呢?defer 不是后进先出吗?但怎么出来个3呢???疯了啊


有疑问加站长微信联系(非本文作者)

入群交流(和以上内容无关):加入Go大咖交流群,或添加微信:liuxiaoyan-s 备注:入群;或加QQ群:692541889

1473 次点击  
加入收藏 微博
4 回复  |  直到 2020-08-26 09:55:21
rwy-aha-QWQ
rwy-aha-QWQ · #1 · 5年之前
func main() {
    for i := 0; i < 3; i++ {
        defer func(i int){
            fmt.Println(i)
        }(i)
    }
}

你对照这个试试就能知道为啥了

jasonwu
jasonwu · #2 · 5年之前
  这是一个闭包,闭包保存或记录了它产生时的外部函数的所有环境。当defer延迟开始执行时,才是真正执行了闭包,此时闭包内部的变量才会进行赋值的操作。
 也就是说,i 这个变量在闭包内的生命周期得到了保证。因此在执行这个闭包的时候,会去外部环境寻找最新的数值!
 此时,变量i的最新值是for循环退出时i的值,为3
slclub
slclub · #3 · 5年之前

无defer 是 0, 1 ,2

有defer defer 定义的所有函数是在 return 之前运行 ,所有函数体内代码之后

有defer 你的函数运行逻辑可以改写成非defer 的代码

func main() {
    K:=0
    for i := 0; i < 3; i++ {
        k = i
    }
    for i:=0; i< 3; i++ {
         func() {
               fmt.Println(k)
         }()
    }
}
tizuiyang
tizuiyang · #4 · 5年之前

楼上正解

添加一条新回复 (您需要 登录 后才能回复 没有账号 ?)
  • 请尽量让自己的回复能够对别人有帮助
  • 支持 Markdown 格式, **粗体**、~~删除线~~、`单行代码`
  • 支持 @ 本站用户;支持表情(输入 : 提示),见 Emoji cheat sheet
  • 图片支持拖拽、截图粘贴等方式上传