百度360必应搜狗淘宝本站头条
当前位置:网站首页 > IT技术 > 正文

Go 1.22终于修复了for循环中的变量问题

wptr33 2025-03-25 18:09 5 浏览

Go 1.21 包含了一个对循环作用域的变更的预览,我们计划在 Go 1.22 中发布此变更,以消除其中一个最常见的 Go 错误。

问题

如果你写过任何数量的 Go 代码,你可能犯过保留循环变量的引用超过其迭代结束的错误,此时它会获得一个你不想要的新值。例如,考虑以下程序:

func main() {
    done := make(chan bool)

    values := []string{"a", "b", "c"}
    for _, v := range values {
        go func() {
            fmt.Println(v)
            done <- true
        }()
    }

    // wait for all goroutines to complete before exiting
    for _ = range values {
        <-done
    }
}

这三个创建的 goroutine 都在打印同一个变量 v,所以它们通常会打印出 "c"、"c"、"c",而不是以某种顺序打印出 "a"、"b" 和 "c"。

Go 的常见问题解答中的条目 "闭包在作为 goroutine 运行时会发生什么?" 给出了这个例子,并指出 "使用闭包与并发时可能会导致一些混淆"。

尽管通常涉及并发,但并非必须如此。这个例子有相同的问题,但没有 goroutine:

func main() {
    var prints []func()
    for i := 1; i <= 3; i++ {
        prints = append(prints, func() { fmt.Println(i) })
    }
    for _, print := range prints {
        print()
    }
}

这种错误已经在许多公司引起了生产问题,包括1619047 - Let's Encrypt: CAA Rechecking bug的问题。在那种情况下,循环变量的意外捕获跨越多个函数,并且更难以注意到:

// authz2ModelMapToPB converts a mapping of domain name to authz2Models into a
// protobuf authorizations map
func authz2ModelMapToPB(m map[string]authz2Model) (*sapb.Authorizations, error) {
    resp := &sapb.Authorizations{}
    for k, v := range m {
        // Make a copy of k because it will be reassigned with each loop.
        kCopy := k
        authzPB, err := modelToAuthzPB(&v)
        if err != nil {
            return nil, err
        }
        resp.Authz = append(resp.Authz, &sapb.Authorizations_MapElement{
            Domain: &kCopy,
            Authz: authzPB,
        })
    }
    return resp, nil
}

这段代码的作者显然清楚地理解了这个通用问题,因为他们对 k 进行了复制,但是 modelToAuthzPB 在构造其结果时使用了 v 中字段的指针,所以循环也需要对 v 进行复制。

已经有工具被开发出来识别这些错误,但分析变量的引用是否超出其迭代的生命周期是很困难的。这些工具必须在误报和漏报之间做出选择。go vet和gopls使用的循环闭包分析器选择了误报,只有在确定存在问题时才报告,而漏报其他情况。其他检查器选择了误报,将正确的代码指责为错误。我们对添加了"x := x"行的开源Go代码进行了分析,期望找到bug修复。然而,我们发现许多不必要的行被添加进去,这表明流行的检查器存在相当高的误报率,但开发人员仍然添加这些行以保持检查器的满意。

我们找到的一对例子特别有启发性:

这个diff是在一个程序中的:

for _, informer := range c.informerMap {
+        informer := informer
         go informer.Run(stopCh)
     }

而这个diff是在另一个程序中的:

  for _, a := range alarms {
+        a := a
         go a.Monitor(b)
     }

其中一个diff是一个bug修复,另一个是一个不必要的更改。除非你了解涉及的类型和函数的更多信息,否则无法确定哪个是哪个。

修复方法

对于Go 1.22,我们计划修改for循环,使这些变量具有每次迭代的作用域,而不是每个循环的作用域。这个改变将修复上述例子,使它们不再是有bug的Go程序;它将终止由此类错误引起的生产问题;并且它将消除需要用户对其代码进行不必要更改的不准确的工具。

为了确保与现有代码的向后兼容性,新的语义将仅适用于在其go.mod文件中声明了go 1.22或更高版本的模块中的包。这个按模块划分的决定使开发人员可以控制在代码库中逐步更新到新语义。还可以使用//go:build行来控制每个文件的决策。

旧代码将继续保持与今天完全一样的含义:修复仅适用于新代码或更新的代码。这将使开发人员能够控制特定包中的语义更改的时间。作为我们前向兼容性工作的结果,Go 1.21将不会尝试编译声明了go 1.22或更高版本的代码。我们在Go 1.20.8和Go 1.19.13的修订版本中包含了具有相同效果的特殊情况,因此,在发布Go 1.22后,依赖新语义编写的代码将永远不会使用旧语义进行编译,除非使用非常旧的不受支持的Go版本。

预览修复方法

Go 1.21 包含了对作用域变更的预览。如果在环境中设置了GOEXPERIMENT=loopvar并使用该选项编译代码,那么新的语义将应用于所有循环(忽略 go.mod 中的 go 行)。例如,要检查在应用新的循环语义到您的包和所有依赖项后,您的测试是否仍然通过:

GOEXPERIMENT=loopvar go test

我们在谷歌的内部 Go 工具链中修复了此模式,并在 2023 年 5 月初开始的所有构建中强制使用该模式,在过去的四个月中,我们在生产代码中没有收到任何问题的报告。

您还可以尝试在 Go Playground 上包含一个 // GOEXPERIMENT=loopvar 的注释,以更好地理解语义,例如在这个程序中(Go Playground - The Go Programming Language)。请注意,这个注释仅适用于 Go Playground

相关推荐

【推荐】一款开源免费、美观实用的后台管理系统模版

如果您对源码&技术感兴趣,请点赞+收藏+转发+关注,大家的支持是我分享最大的动力!!!项目介绍...

Android架构组件-App架构指南,你还不收藏嘛

本指南适用于那些已经拥有开发Android应用基础知识的开发人员,现在想了解能够开发出更加健壮、优质的应用程序架构。首先需要说明的是:AndroidArchitectureComponents翻...

高德地图经纬度坐标批量拾取(高德地图批量查询经纬度)

使用方法在桌面上新建一个index.txt文件,把下面的代码复制进去保存,再把文件名改成index.html保存,双击运行打开即可...

flutter系列之:UI layout简介(flutter ui设计)

简介对于一个前端框架来说,除了各个组件之外,最重要的就是将这些组件进行连接的布局了。布局的英文名叫做layout,就是用来描述如何将组件进行摆放的一个约束。...

Android开发基础入门(一):UI与基础控件

Android基础入门前言:...

iOS的布局体系-流式布局MyFlowLayout

iOS布局体系的概览在我的CSDN博客中的几篇文章分别介绍MyLayout布局体系中的视图从一个方向依次排列的线性布局(MyLinearLayout)、视图层叠且停靠于父布局视图某个位置的框架布局(M...

TDesign企业级开源设计系统越发成熟稳定,支持 Vue3 / 小程序

TDesing发展越来越好了,出了好几套组件库,很成熟稳定了,新项目完全可以考虑使用。...

WinForm实现窗体自适应缩放(winform窗口缩放)

众所周知,...

winform项目——仿QQ即时通讯程序03:搭建登录界面

上两篇文章已经对CIM仿QQ即时通讯项目进行了需求分析和数据库设计。winform项目——仿QQ即时通讯程序01:原理及项目分析...

App自动化测试|原生app元素定位方法

元素定位方法介绍及应用Appium方法定位原生app元素...

61.C# TableLayoutPanel控件(c# tabcontrol)

摘要TableLayoutPanel在网格中排列内容,提供类似于HTML元素的功能。TableLayoutPanel控件允许你将控件放在网格布局中,而无需精确指定每个控件的位置。其单元格...

想要深入学习Android性能优化?看完这篇直接让你一步到位

...

12个python数据处理常用内置函数(python 的内置函数)

在python数据分析中,经常需要对字符串进行各种处理,例如拼接字符串、检索字符串等。下面我将对python中常用的内置字符串操作函数进行介绍。1.计算字符串的长度-len()函数str1='我爱py...

如何用Python程序将几十个PDF文件合并成一个PDF?其实只要这四步

假定你有一个很无聊的任务,需要将几十个PDF文件合并成一个PDF文件。每一个文件都有一个封面作为第一页,但你不希望合并后的文件中重复出现这些封面。即使有许多免费的程序可以合并PDF,很多也只是简单的将...

Python入门知识点总结,Python三大数据类型、数据结构、控制流

Python基础的重要性不言而喻,是每一个入门Python学习者所必备的知识点,作为Python入门,这部分知识点显得很庞杂,内容分支很多,大部分同学在刚刚学习时一头雾水。...