浅谈go中defer的一个隐藏功能

作者:KevinYan 时间:2024-04-28 09:13:07 

在开始使用Go进行编码时,Defer是要关注的一个很重要的特性。它非常简单:在任何函数中,给其他函数的调用加上前缀 defer以确保该函数在外部函数退出之前立即执行,即使外部函数出现异常被中断,该延迟函数也将运行。

但是,你还可以使用defer在任何函数开始后和结束前执行配对的代码。这个隐藏的功能在网上的教程和书籍中很少提到。要使用此功能,需要创建一个函数并使它本身返回另一个函数,返回的函数将作为真正的延迟函数。在 defer 语句调用父函数后在其上添加额外的括号来延迟执行返回的子函数如下所示:


func main() {
 defer greet()()
 fmt.Println("Some code here...")
}

func greet() func() {
 fmt.Println("Hello!")
 return func() { fmt.Println("Bye!") } // this will be deferred
}

输出以下内容:

Hello!
Some code here...
Bye!

父函数返回的函数将是实际的延迟函数。父函数中的其他代码将在函数开始时(由 defer 语句放置的位置决定)立即执行。

这为开发者提供了什么能力?因为在函数内定义的匿名函数可以访问完整的词法环境(lexical environment),这意味着在函数中定义的内部函数可以引用该函数的变量。在下一个示例中看到的,参数变量在measure函数第一次执行和其延迟执行的子函数内都能访问到:


func main() {
 example()
 otherExample()
}

func example(){
 defer measure("example")()
 fmt.Println("Some code here")
}

func otherExample(){
 defer measure("otherExample")()
 fmt.Println("Some other code here")
}

func measure(name string) func() {
 start := time.Now()
 fmt.Printf("Starting function %s\n", name)
 return func(){ fmt.Printf("Exiting function %s after %s\n", name, time.Since(start)) }
}

输出以下内容:

Starting example
Some code here
Exiting example after 0s
Starting otherExample
Some other code here
Exiting otherExample after 0s

此外函数命名的返回值也是函数内的局部变量,所以上面例子中的measure函数如果接收命名返回值作为参数的话,那么命名返回值在延迟执行的函数中访问到,这样就能将measure函数改造成记录入参和返回值的工具函数。

下面的示例是引用《go 语言程序设计》中的代码段:


func bigSlowOperation() {
 defer trace("bigSlowOperation")() // don't forget the extra parentheses
 // ...lots of work…
 time.Sleep(10 * time.Second) // simulate slow
 operation by sleeping
}
func trace(msg string) func() {
 start := time.Now()
 log.Printf("enter %s", msg)
 return func() {
   log.Printf("exit %s (%s)", msg,time.Since(start))
 }
}

可以想象,将代码延迟在函数的入口和出口使用是非常有用的功能,尤其是在调试代码的时候。

来源:https://segmentfault.com/a/1190000021239576

标签:go,defer
0
投稿

猜你喜欢

  • python扫描proxy并获取可用代理ip的实例

    2023-07-29 16:42:50
  • Python数值求解微分方程方法(欧拉法,隐式欧拉)

    2023-06-29 10:45:29
  • Python日期的加减等操作的示例

    2021-10-06 14:28:14
  • 微信小程序之多文件下载的简单封装示例

    2023-10-19 21:10:06
  • python安装sklearn模块的方法详解

    2023-08-01 17:23:48
  • golang实现数组分割的示例代码

    2024-05-08 10:23:41
  • Python切片工具pillow用法示例

    2021-12-08 21:17:26
  • Python 照片人物背景替换的实现方法

    2021-10-21 08:45:02
  • 解决运行出现'dict' object has no attribute 'has_key'问题

    2021-06-14 06:58:48
  • Python实现图像去噪方式(中值去噪和均值去噪)

    2023-04-15 15:38:13
  • Python3网络爬虫之使用User Agent和代理IP隐藏身份

    2022-03-05 13:13:07
  • Python学习笔记之os模块使用总结

    2023-05-12 07:46:21
  • python lambda表达式在sort函数中的使用详解

    2021-09-22 02:07:56
  • 浅谈Python小波分析库Pywavelets的一点使用心得

    2023-03-11 22:55:23
  • PHP 字符串编码截取函数(兼容utf-8和gb2312)

    2024-04-29 13:56:37
  • Python编程源码报错解决方法总结经验分享

    2023-11-16 07:49:46
  • python基础教程项目五之虚拟茶话会

    2022-10-25 02:08:47
  • Python中用Ctrl+C终止多线程程序的问题解决

    2021-10-22 03:22:18
  • Python列表删除重复元素与图像相似度判断及删除实例代码

    2021-02-21 05:28:58
  • JavaScript引入方式深入解读

    2024-04-25 13:12:29
  • asp之家 网络编程 m.aspxhome.com