迹忆客 专注技术分享

当前位置:主页 > 学无止境 > 编程语言 > Go >

Go的匿名返回值和命名返回值

作者:迹忆客 最近更新:2022/10/21 浏览次数:

引言

Go的函数返回可以说是别出心裁,既可以用匿名返回值,也可以用命名返回值,虽然这两种方式都可以完成相同的功能,但是这里边还是有一些坑,尤其是在搭配defer的时候,返回的值可能不是你想的那样,很容易产生不太好排查的问题


举个例子

func f1() int {
    k := 0
    for i := 0; i < 3; i++ {
        defer func() {k++}()
    }
    return k
}

func f2() (k int) {
    for i := 0; i < 3; i++ {
        defer func() {k++}()
    }
    return
}

func f3() (k int) {
    s := 5
    for i := 0; i < 3; i++ {
        defer func() {k++}()
    }
    return s
}

func f4() (k int) {
    for i := 0; i < 3; i++ {
        defer func(k int) {k++}(k)
    }
    return
}

思考一下,这四个函数,分别返回的值都是多少。分别是0,3,8,0。看起来差不多的函数,返回来的值却大不一样。


解释

关键点

  • return语句 这个return v 语句,这个语句编译后分成多步进行,
    • 赋值
      • 如果返回值是匿名值的话,做返回值=v,也就是找一个临时变量保存住这个返回值
      • 如果不是匿名返回值
        • return后边有值v:那还是做返回值=v,返回值就是函数定义的那个命名返回值
        • return后边没有值: 默认就是返回这个命名返回值
    • 如果有defer,后进先出的方式执行defer
    • 返回返回值
  • defer的值拷贝和值引用
    • 如果没有定义传参,那就是用defer外的同名变量,而且是采用值引用的方式
    • 如果定义了传参,那就是值拷贝,不影响defer外边的变量

函数解释

  • f1 由于是匿名返回值,需要先试用临时变量(tmp)保存返回值,tmp=k,不管后边对k做怎样的操作,那都是k的变化,影响不到tmp,然后最后返回tmp,也就是0

  • f2 返回值为命名返回值,而且return后边没有显式返回数据,默认就是返回k,不需要提前赋值,后续的defer中没有采用值传递的方式传参,是地址引用,因此,始终在操作这个k,最后返回k时,肯定是最新的k,也就是3

  • f3 返回值也是命名返回值,return却显式的返回了s,所以需要有赋值动作,也就是k=s(为什么不是tmp,因为不需要临时值,返回值已经命名为k了),后边的defer还是地址引用传参,所有的操作对k均产生了影响,返回是当然是k的最新值,也就是8

  • f4 虽然也是命名返回值,return后边也没有显式的返回其他值,默认就是返回k,但是注意看defer却是用的值传递,应为定义了参数,也就是defer在入栈的时候,已经将k复制成了另外的变量(比如tmp1,tmp2,tmp3)分别跟着入了栈,defer执行时,显然不会影响真正的返回值k,毕竟地址都不是一个了。所以最后的返回值是0。

想想如果在k++后边加个fmt.Println(“—k–”,k),会是多少?

func f4() (k int) {
    for i := 0; i < 3; i++ {
        defer func(k int) {
      k++
      fmt.Println("---k--", &k, k)
    }(k)
    }
    return
}

结果如下所示

---k-- 0xc000124048 1
---k-- 0xc000124050 1
---k-- 0xc000124058 1

答案:输出三次,都是1,而且地址都不一样。原因很简单,上边说的defer会有三次入栈,且每次入栈的时候分别新创建了三个内部变量(所以地址不一样),并都赋值成了0,也就是看着是k,实际早已经不是一个k了,所以每次给0++ ,那只能是输出0了。

转载请发邮件至 1244347461@qq.com 进行申请,经作者同意之后,转载请以链接形式注明出处

本文地址:

相关文章

使用 C 语言中的 goto 语句

发布时间:2023/05/07 浏览次数:79 分类:C语言

本文介绍了如何在 C 语言中使用 goto 语句。使用 goto 语句在 C 语言中实现循环 goto 关键字是 C 语言的一部分,它提供了一个做无条件跳转的结构。

Django 中的 Slug

发布时间:2023/05/04 浏览次数:173 分类:Python

本篇文章旨在定义一个 slug 以及我们如何使用 slug 字段在 Python 中使用 Django 获得独特的帖子。

Django ALLOWED_HOSTS 介绍

发布时间:2023/05/04 浏览次数:181 分类:Python

本文展示了如何创建您的 Django 网站,为公开发布做好准备,如何设置 ALLOWED_HOSTS 以及如何在使用 Django 进行 Web 部署期间修复预期的主要问题。

Django 中的 Select_related 方法

发布时间:2023/05/04 浏览次数:129 分类:Python

本文介绍了什么是查询集,如何处理这些查询以及我们如何利用 select_related() 方法来过滤 Django 中相关模型的查询。

在 Django 中上传媒体文件

发布时间:2023/05/04 浏览次数:198 分类:Python

在本文中,我们简要介绍了媒体文件以及如何在 Django 项目中操作媒体文件。

Django 返回 JSON

发布时间:2023/05/04 浏览次数:106 分类:Python

在与我们的讨论中,我们简要介绍了 JSON 格式,并讨论了如何借助 Django 中的 JsonResponse 类将数据返回为 JSON 格式。

在 Django 中创建对象

发布时间:2023/05/04 浏览次数:59 分类:Python

本文的目的是解释什么是模型以及如何使用 create() 方法创建对象,并了解如何在 Django 中使用 save() 方法。

扫一扫阅读全部技术教程

社交账号
  • https://www.github.com/onmpw
  • qq:1244347461

最新推荐

教程更新

热门标签

扫码一下
查看教程更方便