Golang命名返回值

介绍

Golang 语言支持命名返回值,它与使用普通(匿名)返回值不同的是,命名返回值会被视为定义在函数顶部的变量,并且在使用 return 语句返回时,不再必须在其后面指定参数名,也就是支持“裸”返回。

而使用普通返回值时,使用 return 语句返回时,需要在其后面指定与普通返回值相同类型的参数名。

实际上,命名返回值和普通返回值都有其适用的场景,本文我们介绍 Golang 语言函数或方法使用命名返回值和普通返回值各自的“好处”与“坏处”。

命名返回值

使用命名返回值的“好处”是可以提升代码可读性,读者朋友们试想一下,当函数或方法有多个返回值时,尤其是函数体中代码比较长的函数或方法,如果我们使用普通返回值,那么我们想要知道返回值的含义,就需要先阅读函数体中完整代码。

而如果使用具有实际含义的命名返回值,我们只需要阅读函数或方法的签名,就可以知道其含义,甚至可以把它们作为文档使用。

但是,命名返回值也不是没有“坏处”,如果函数体内有变量与命名返回值同名,那么命名返回值会被覆盖,所以我们也需要注意避免“踩坑”。

普通返回值

普通(匿名)返回值的“好处”是简洁,当我们写一些简短函数或方法时,使用普通返回值可以使代码更加简洁,在 Golang 语言官方标准库中,有很多使用普通返回值的函数或方法。

但是如果返回值是指针类型时,使用普通返回值,就会使我们函数体中的代码变得不优雅,比如以下这段示例代码。

1
2
3
4
5
6
7
8
func c() *int {
i := 0
return &i
}

func d() (i *int) {
return
}

当然这里列举的代码片段是个极端示例,我们在编写 Golang 代码时,也并不会这么使用。

还有就是在编写函数体代码比较长的函数时,使用普通返回值的代码,其可读性比不上使用命名返回值的代码。

踩坑

defer 在命名返回值和普通返回值的函数或方法中,返回的结果不一样。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
func main() {
f := fmt.Println
f(a())
f(b())
}

func a() int {
i := 0
defer func() {
i += 1
fmt.Println("a defer:", i)
}()
return i
}

func b() (i int) {
i = 0
defer func() {
i += 1
fmt.Println("b defer:", i)
}()
return i
}

输出结果:

1
2
3
4
a defer: 1
0
b defer: 1
1

阅读上面这段代码,我们可以发现使用普通返回值的函数 a(),返回结果是 0。使用命名返回值的函数 b(),返回结果是 1

我们在之前的文章中,也单独介绍过 defer 。在这里我们简述一下,当我们使用 defer 调用一个函数时,该函数的执行,被推迟到周围函数返回的那一刻,要么是因为周围的函数执行了 return 语句,要么是因为相应的 goroutine 崩溃。

在函数 a() 中,因为我们没有使用命名返回值,所以返回结果 return i,其中 i 是一个静态值,即使我们在 defer 调用的函数中给变量 i 执行 +1 操作,返回结果中的变量 i 是不可访问的,所以也不会修改返回结果中的变量 i

在函数 b() 中,因为我们使用命名返回值,所以变量 i 已被分配,并且被初始化为类型零值。即使 defer 调用的函数在返回结果之后执行,返回结果中的变量 i 仍然是可以被访问的,所以其值仍然可以被修改。