阿里云-云小站(无限量代金券发放中)
【腾讯云】云服务器、云数据库、COS、CDN、短信等热卖云产品特惠抢购

简单探讨Golang中defer预计算参数

73次阅读
没有评论

共计 1811 个字符,预计需要花费 5 分钟才能阅读完成。

导读 在 golang 当中 defer 代码块会在函数调用链表中增加一个函数调用, 下面这篇文章主要给大家介绍了关于 Golang 中 defer 预计算参数的相关资料, 文中通过实例代码介绍的非常详细, 需要的朋友可以参考下
什么是 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

到此这篇关于 Golang 中 defer 预计算参数的文章就介绍到这了。

阿里云 2 核 2G 服务器 3M 带宽 61 元 1 年,有高配

腾讯云新客低至 82 元 / 年,老客户 99 元 / 年

代金券:在阿里云专用满减优惠券

正文完
星哥玩云-微信公众号
post-qrcode
 0
星锅
版权声明:本站原创文章,由 星锅 于2024-07-25发表,共计1811字。
转载说明:除特殊说明外本站文章皆由CC-4.0协议发布,转载请注明出处。
【腾讯云】推广者专属福利,新客户无门槛领取总价值高达2860元代金券,每种代金券限量500张,先到先得。
阿里云-最新活动爆款每日限量供应
评论(没有评论)
验证码
【腾讯云】云服务器、云数据库、COS、CDN、短信等云产品特惠热卖中