
本文旨在阐明go语言中直接调用函数与使用函数指针调用函数时,在闭包和状态维护方面的差异。通过一个斐波那契数列的示例,深入解析了两种调用方式导致的不同结果,并解释了其背后的原因,帮助读者更好地理解go语言中函数作为一等公民的特性。
Go语言中函数调用与函数指针的差异详解
在Go语言中,函数是一等公民,可以像其他类型一样被赋值给变量、作为参数传递给其他函数,或者作为返回值从函数中返回。理解函数调用与函数指针之间的差异,对于编写高效且易于维护的Go代码至关重要。本文将通过一个斐波那契数列的生成器示例,深入探讨这两种方式的不同之处。
斐波那契数列生成器示例
首先,我们定义一个名为 fibonacci 的函数,该函数返回一个匿名函数,该匿名函数负责生成斐波那契数列的下一个数字。
package main import "fmt" // fibonacci is a function that returns // a function that returns an int. func fibonacci() func() int { previous := 0 current := 1 return func () int{ current = current+previous previous = current-previous return current } }
在这个例子中,fibonacci 函数创建了一个闭包。闭包是由函数及其相关的引用环境组合而成的实体。在这个例子中,返回的匿名函数“记住”了 previous 和 current 变量,即使 fibonacci 函数已经执行完毕。
立即学习“go语言免费学习笔记(深入)”;
直接调用与函数指针调用
现在,让我们比较两种不同的调用方式:
方式一:错误的示例
func main() { f := fibonacci for i := 0; i < 10; i++ { fmt.Println(f()()) } }
这段代码的输出是 10 个 1。这是因为 f := fibonacci 将 fibonacci 函数本身赋值给变量 f,而不是调用 fibonacci 函数并将其返回值(即生成器函数)赋值给 f。因此,在循环的每次迭代中,f()() 实际上是:
- f():调用 fibonacci 函数,创建一个新的斐波那契数列生成器。
- ():立即调用新创建的生成器,并获取其返回的第一个值(始终为 1)。
每次循环都创建一个新的生成器,只使用其第一个值,然后丢弃该生成器。因此,我们总是得到 1。
方式二:正确的示例
func main() { f := fibonacci() for i := 0; i < 10; i++ { fmt.Println(f()) } }
这段代码会正确地输出斐波那契数列的前 10 个数字。这是因为 f := fibonacci() 首先调用 fibonacci 函数,并将其返回值(即生成器函数)赋值给变量 f。因此,在循环的每次迭代中,f() 都会调用同一个生成器,从而保持了 previous 和 current 变量的状态,并生成正确的斐波那契数列。
总结与注意事项
- 理解闭包: 闭包是Go语言中强大的特性,允许函数访问和修改其定义时的环境中的变量。
- 函数指针与函数调用: 区分将函数赋值给变量(函数指针)和调用函数并将其返回值赋值给变量。
- 状态维护: 如果需要维护状态,确保在循环中使用同一个生成器实例。
- 避免重复创建: 避免在循环中重复创建生成器,除非确实需要每次都从头开始生成。
通过这个例子,我们深入理解了Go语言中函数调用与函数指针的不同行为,以及闭包在状态维护中的作用。在实际编程中,根据具体需求选择合适的调用方式,可以编写出更高效、更易于维护的代码。


