文档章节

Go 关键字 defer 的一些坑

云迹
 云迹
发布于 2017/09/03 02:23
字数 724
阅读 12
收藏 0

什么是 defer?如何理解 defer 关键字?Go 中使用 defer 的一些坑。

defer 意为延迟,在 golang 中用于延迟执行一个函数。它可以帮助我们处理容易忽略的问题,如资源释放、连接关闭等。但在实际使用过程中,有一些需要注意的地方(坑),下面我们一一道来。

一些结论

首先,我们来了解 defer 的一些结论:

1、若函数中有多个 defer,其执行顺序为 先进后出,可以理解为栈。

GOpackage main

import "fmt"

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

Output:
4
3
2
1
0

2、return 会做几件事:

  1. 给返回值赋值
  2. 调用 defer 表达式
  3. 返回给调用函数
GOpackage main

import "fmt"

func main() {
    fmt.Println(increase(1))
}

func increase(d int) (ret int) {
  defer func() {
    ret++
  }()

  return d
}
  
Output:
2

3、若 defer 表达式有返回值,将会被丢弃。

更多请参考官方文档

闭包与匿名函数

匿名函数:没有函数名的函数。
闭包:可以使用另外一个函数作用域中的变量的函数。

在实际开发中,defer 的使用经常伴随着闭包与匿名函数的使用。小心踩坑哦:

GOpackage main

import "fmt"

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

Output:
5
5
5
5
5

解释一下,defer 表达式中的 i 是对 for 循环中 i 的引用。到最后,i 加到 5,故最后全部打印 5。

如果将 i 作为参数传入 defer 表达式中,在传入最初就会进行求值保存,只是没有执行延迟函数而已。

GOfor i := 0; i < 5; i++ {
    defer func(idx int) {
        fmt.Println(idx)
    }(i) // 传入的 i,会立即被求值保存为 idx
}

巩固一下

为了巩固一下上面的知识点,我们来思考几个例子。

例1:

GOfunc f() (result int) {
    defer func() {
        result++
    }()
    return 0
}

例2:

GOfunc f() (r int) {
    t := 5
    defer func() {
        t = t + 5
    }()
    return t
}

例3:

GOfunc f() (r int) {
    defer func(r int) {
        r = r + 5
    }(r)
    return 1
}

有没有得出结果?例1的答案不是 0,例2的答案不是 10,例3的答案也不是 6。

例1,比较简单,参考结论2,将 0 赋给 result,defer 延迟函数修改 result,最后返回给调用函数。正确答案是 1。

例2,defer 是在 t 赋值给 r 之后执行的,而 defer 延迟函数只改变了 t 的值,r 不变。正确答案 5。

例3,这里将 r 作为参数传入了 defer 表达式。故 func (r int) 中的 r 非 func f() (r int) 中的 r,只是参数命名相同而已。正确答案 1。

参考文档

[1] https://tiancaiamao.gitbooks.io/go-internals/content/zh/03.4.html

[2] http://golang.org/ref/spec#defer_statements

本文链接:https://deepzz.com/post/how-to-use-defer-in-golang.html参与评论 »

--EOF--

发表于 2017-08-27 02:08:00。

本站使用「署名 4.0 国际」创作共享协议,转载请注明作者及原网址。更多说明 »

本文转载自:https://deepzz.com/post/how-to-use-defer-in-golang.html

共有 人打赏支持
云迹
粉丝 4
博文 43
码字总数 4969
作品 0
成都
程序员
Golang defer 使用时的坑

defer是golang语言中的关键字,用于资源的释放,会在函数返回之前进行调用。一般采用如下模式: f,err := os.Open(filename)if err != nil { }defer f.Close() 如果有多个defer表达式,调用顺...

徐学良
2016/07/20
90
0
defer函数参数求值简要分析

defer函数参数求值简要分析 Tony Bai2018-03-231 阅读 FunctiondeferGo 一. 引子 书接上文,在发表了 《对一段Go语言代码输出结果的简要分析》 一文之后,原问题提出者又有了新问题,这是一个...

Tony Bai
03/23
0
0
script的加载方式与执行

script一般是阻塞式加载的,H5新增了、和特性,可用于异步加载/延迟执行: async 属性是指当这个 script 可用时,就异步执行它 defer 属性是指当页面被解析完毕后,才能执行 如果以上两个属性...

一点灵犀
2016/06/16
18
0
Golang中defer、return、返回值之间执行顺序的坑

Go语言中延迟函数defer充当着 try...catch 的重任,使用起来也非常简便,然而在实际应用中,很多gopher并没有真正搞明白defer、return和返回值之间的执行顺序,从而掉进坑中,今天我们就来揭...

henrylee2cn
2015/09/13
7.5K
1
Go基础编程:延迟调用defer

1 defer作用 关键字 defer ⽤于延迟一个函数或者方法(或者当前所创建的匿名函数)的执行。注意,defer语句只能出现在函数或方法的内部。 defer语句经常被用于处理成对的操作,如打开、关闭、...

tennysonsky
2017/12/29
0
0

没有更多内容

加载失败,请刷新页面

加载更多

下一页

180.mariadb 主从复制

参考:https://blog.csdn.net/chengxuzaza/article/details/62042920 睡觉睡觉,明天写 1.效果 当主库中数据有变化的时候,从库就自动同步 2. 环境要求 至少两台 linux服务器 (教程:https...

Lucky_Me
8分钟前
0
0
erlng file id3v1 id3v1.1

%% ---%% Excerpted from "Programming Erlang",%% published by The Pragmatic Bookshelf.%% Copyrights apply to this code. It may not be used to create training material, %% ......

xueyuse0012
9分钟前
0
0
RabbitMq的安装

环境Centos6.5 32位 JDK 1.7.8 Jdk的卸载 rpm -qa|grep jdk yum –y remove 上边的安装包 JDK的安装 Rpm –ivh jdk安装包 配置环境变量 export JAVA_BIN=/usr/java/jdk1.7.0_80/bin export J......

DemonsI
13分钟前
0
0
http和https协议

HTTPS全称为Hypertext Transfer Protocol over Secure Socket Layer,中文含义为“超文本传输协议在安全加密字层”,简单来说就是加密数据传输,通俗的说就是安全连接。 HTTPS安全超文本传输...

寰宇01
19分钟前
0
0
vue内引入语音播报功能

在vue项目中引入语音播报,使用的科大讯飞语音接入, 具体思路为每次接收到语音信息后存入一个数组,然后监听这个数组,开始冲第一个索引播放,并且同时根据vuex getter 来动态删减数量 给a...

originDu
28分钟前
0
0

没有更多内容

加载失败,请刷新页面

加载更多

下一页

返回顶部
顶部