一道正确率只有15%的命名返回值和闭包的问题

前言

哈喽,大家好,我是asong。今天新注册了twitter,在里面没事瞎逛的时候,发现了一道有意思的题,他是由Redhat的首席工程师、Prometheus开源项目维护者 Bartłomiej Płotka 发出的,经调查显示,这道题的正确率只有15.2%,惨目忍睹,接下来我们就一起来看一下这道题~

原文地址:https://twitter.com/bwplotka/status/1495002204163678211

题目

下面这段代码输出结果是多少?

func aaa() (done func(), err error) {
 return func() { print("aaa: done") }, nil
}

func bbb() (done func(), _ error) {
 done, err := aaa()
 return func() { print("bbb: surprise!"); done() }, err
}

func main() {
 done, _ := bbb()
 done()
}

来吧,朋友们,选出你的答案吧~

总共四个答案:

题目解析

正确答案是【C】:永远不会结束,你答对了吗?

这道题考查的点就是命名返回值+闭包,把上面的代码换成等效的匿名返回值代码你就明白了:

func aaa() (func(), error) {
 var done func()
 done = func() {
  print("aaa: done")
 }
 return done, nil
}

func bbb() (func(), error) {
 var done func()
 done, err := aaa()
 done = func() {
  print("bbb: surprise!");
  done()
 }
 return done, err
}

func main() {
 done, _ := bbb()
 done()
}

这其实是Go语言设计上一个feature,当Go语言的返回值赋给我们特殊的"返回参数"时,如果它们被命名了,在return之后,我们可以在函数主体完成后的任何执行过程中引用那些带有这些名称的值,在defer或闭包中一样。

我们在说回这道题,在bbb()函数内我们使用了命名返回值done func(), _ error,使用短变量声明done, err := aaa()接收aaa()的返回值,这里变量done并不是一个新变量,这就要说到Go语言的短变量声明的语法糖了,在多变量声明中,如果其中一个变量是新的,可以使用 := 声明,编译器会进行类型推断和赋值,已经声明的变量不会重新声明,直接在原变量上赋值;之后我们return的是一个闭包函数,闭包里的done值并不会被提前解析,在bbb()函数结束后,实际对应的代码就成了这样,变成了递归。

done = func() {
  print("bbb: surprise!");
  done()
}

如果我们把代码在改成这样:

func bbb() (func(), error) {
 var done func()
 done, err := aaa()
 return func() {
  print("bbb: surprise!");
  done()
 }, err
}

答案就是【B】:bbb: surprise!aaa: done

总结

一道看似简单的题,其中蕴涵的知识点确有很多,这就说明了解设计原理是多么的重要,Go语言资深工程师的路上任重道远呀~。

好啦,本文到这里就结束了,我是asong,我们下期见。

创建了读者交流群,欢迎各位大佬们踊跃入群,一起学习交流。入群方式:关注公众号获取。更多学习资料请到公众号领取。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值