简单聊聊Golang中defer预计算参数

作者:蓝色记忆 时间:2023-07-22 03:55:09 

什么是defer

defer用来声明一个延迟函数,把这个函数放入到一个栈上, 当外部的包含方法return之前,返回参数到调用方法之前调用,也可以说是运行到最外层方法体的"}"时调用。我们经常用他来做一些资源的释放,比如关闭io操作

func doSomething(fileName string) {
   file,err := os.Open(fileName)
   if err != nil {
   panic(err)
   }
   defer file.Close()
}

defer 可以保证方法可以在外围函数返回之前调用。有点像其他言的 try finally

try{
}finally{
}

Go语言defer预计算参数

Go 语言中所有的函数调用都是传值的,虽然 defer 是关键字,但是也继承了这个特性。假设我们想要计算 main 函数运行的时间,可能会写出以下的代码:

package main
import (
"fmt"
"time"
)

func main() {
startedAt := time.Now()
defer fmt.Println(time.Since(startedAt))
time.Sleep(time.Second) //休眠一秒
}

结果是:

D:\workspace\go\src\test>go run main.go
0s 

运行结果并不符合我们的预期,这个现象背后的原因是什么呢?经过分析,我们会发现调用 defer 关键字会立刻拷贝函数中引用的外部参数,所以 time.Since(startedAt) 的结果不是在 main 函数退出之前计算的,而是在 defer 关键字调用时计算的【defer入栈的时候】,最终导致上述代码输出 0s

我们再来看个简单例子来说明上述解释:

package main
import (
"fmt"
)

func main() {
i := 1
defer fmt.Println(test(i))
i = 100
}

func test(i int) int {
i = i + 1
return i
}

D:\workspace\go\src\test>go run main.go
2

当代码运行到defer fmt.Println(test(i))的时候,会把defer右边最外层函数的参数计算完毕,并传递进函数里,但不会执行函数体的代码直到包裹defer的函数返回。我们先看会把defer右边最外层函数的参数计算完毕,并传递进函数里这句话,对应例子就是先把test(i)算出来,此时i=1,计算test(1)得2,然后fmt.Println(2)入栈,等到最后程序运行完了再运行defer结果就是2(但不会执行函数体的代码直到包裹defer的函数返回)。

我们再来看一个例子与匿名函数结合:

package main
import (
"fmt"
)

func main() {
i := 1
defer func() {
fmt.Println(test(i))
}()
i = 100
}

func test(i int) int {
i = i + 1
return i
}

结果:

D:\workspace\go\src\test>go run main.go
101  

使用匿名函数,结果是101,相当于i给到test方法的是100,那为什么呢?还是那句话:但不会执行函数体的代码直到包裹defer的函数返回

也就是说他会把整个{ fmt.Println(test(i)) }()函数体入栈,等到最后程序运行完了再运行defer,此时的i是100,运行test后就是101了。

所以你要解决第一个打印为0s的问题,你就可以使用匿名函数来解决,如下:

package main
import (
"fmt"
"time"
)

func main() {
startedAt := time.Now()
defer func() {
fmt.Println(time.Since(startedAt))
}()
time.Sleep(time.Second) //休眠一秒
}

结果:

D:\workspace\go\src\test>go run main.go
1.0152825s

来源:https://juejin.cn/post/7075529260315705380/

标签:golang,defer,预计算
0
投稿

猜你喜欢

  • Python 将代码转换为可执行文件脱离python环境运行(步骤详解)

    2023-05-02 13:29:31
  • Python如何实现大型数组运算(使用NumPy)

    2023-05-11 21:49:05
  • VS2003 SP1发布

    2024-06-05 09:26:10
  • python基于 Web 实现 m3u8 视频播放的实例

    2022-06-15 22:16:40
  • Django实战之用户认证(用户登录与注销)

    2023-03-23 16:52:26
  • python k-近邻算法实例分享

    2022-03-26 14:47:17
  • 鼠年发几张可爱老鼠的表情gif

    2008-01-29 12:50:00
  • 像聪明女孩穿衣服那样设计网页文字

    2007-11-06 16:45:00
  • python numpy数组中的复制知识解析

    2023-08-10 14:41:33
  • 在asp.net中KindEditor编辑器的使用方法小结

    2023-03-11 21:13:41
  • golang flag简单用法

    2024-05-08 10:23:48
  • python多进程读图提取特征存npy

    2022-09-05 11:38:06
  • vue项目中如何引入cesium

    2024-05-28 15:52:29
  • Python开发.exe小工具的详细步骤

    2021-07-11 10:49:30
  • django 基于中间件实现限制ip频繁访问过程详解

    2022-06-24 08:38:43
  • python代码实现学生信息管理系统

    2023-02-06 18:31:34
  • Python 中的集合和字典

    2021-03-18 22:53:30
  • 深入了解Django View(视图系统)

    2022-07-18 14:47:13
  • Python 编程语言详细介绍

    2022-08-21 08:39:27
  • js树插件zTree获取所有选中节点数据的方法

    2024-05-02 16:20:55
  • asp之家 网络编程 m.aspxhome.com