在 Go 中延迟函数执行时,了解评估行为至关重要。正如规范中所述,当遇到 defer 语句时,会立即计算函数的参数和值。这可能会导致意外结果,特别是在将 defer 与函数内修改的变量一起使用时。
考虑以下函数:
func printNumbers() { var x int defer fmt.Println(x) for i := 0; i < 5; i++ { x++ } }
该函数的目的是在函数执行结束时打印 x (5) 的最终值。但是,由于在 defer 语句期间立即评估 x,因此将打印零。
为了解决此问题,存在几种替代解决方案:
1.匿名函数:
不要推迟原始函数,而是创建一个捕获并打印 x 的当前值的匿名函数。
func printNumbers() { var x int defer func() { fmt.Println(x) }() for i := 0; i < 5; i++ { x++ } }
2.指针:
使用指向 x 的指针来推迟实际值。由于在 defer 语句期间仅评估地址,因此将打印 x 的最终值。
func printNumbers() { var x int defer fmt.Println(&x) for i := 0; i < 5; i++ { x++ } } func Print(i *int) { fmt.Println(*i) }
3.自定义类型:
创建一个实现 fmt.Stringer 的自定义类型。这允许类型定义如何打印,确保显示正确的值。
type MyInt int func (m *MyInt) String() string { return strconv.Itoa(int(*m)) } func printNumbers() { var x MyInt defer fmt.Println(&x) for i := 0; i < 5; i++ { x++ } }
4.切片:
切片是引用底层数据结构的描述符。通过延迟包含变量的切片,将打印最终修改的值。
func printNumbers() { x := []int{0} defer fmt.Println(x) for i := 0; i < 5; i++ { x[0]++ } }
了解延迟函数执行的立即评估行为对于有效使用 defer 至关重要。提供的解决方案演示了确保在推迟执行函数时打印预期值的各种方法。
以上是Go 的'defer”语句如何处理变量求值,对于意外行为有哪些解决方法?的详细内容。更多信息请关注PHP中文网其他相关文章!