理解 Go 中“defer”的复杂性
Go 中的 Defer 是一个强大的关键字,可以让你推迟函数的执行调用直到封闭函数返回。然而,在使用 defer 时,掌握函数参数和返回值的求值顺序至关重要。
求值顺序和 Defer
根据 Defer_statements 的文档,每次执行“defer”语句时,“调用的函数值和参数都会照常评估并重新保存,但不会调用实际函数。”这意味着当您调用延迟函数时,首先评估函数参数,然后在函数返回期间调用实际函数。
场景
让我们深入研究一下进入一个特定的示例,您无法根据用户的年龄正确计算票价。您的代码定义了对 printTicket 的延迟调用,参数为 TicketPrice。您期望根据提供的年龄设置ticketPrice并返回适当的票价。
问题
问题的出现是因为您试图通过printTicket 的未初始化“ticketPrice”。当在 main 返回期间执行对 printTicket 的延迟调用时,未初始化的“ticketPrice”将导致默认值为零,导致仅打印 9.99 的价格。
解决方案
要解决此问题,您应该在推迟调用 printTicket 之前将“ticketPrice”初始化为正确的值。或者,您可以使用修改后的函数调用语法,允许您通过引用传递“ticketPrice”,确保延迟调用获取更新的值。
示例
这是代码的修改版本,演示了 defer 的正确用法:
<code class="go">package main import "fmt" func printTicket(age int) float64 { fmt.Println("...order is 2...") switch { case age <= 13: return 9.99 case age > 13 && age < 65: return 19.99 default: return 12.99 } } func main() { var age int defer fmt.Println("...order is 4...Your age is:", getAge(&age)) // Modify the calling syntax here: var ticketPrice = 0.0 defer fmt.Println("...order is 3...Your ticket price is:", printTicket(age, &ticketPrice)) } func getAge(age *int) int { fmt.Println("...order is 1...") fmt.Print("Enter age=") fmt.Scanln(age) return *age }</code>
通过这些修改,“ticketPrice”变量将在“printTicket”调用之前正确初始化,确保正确的票价计算并打印。
以上是Go 中的 defer 如何处理函数参数?的详细内容。更多信息请关注PHP中文网其他相关文章!