码农知识堂 - 1000bd
  •   Python
  •   PHP
  •   JS/TS
  •   JAVA
  •   C/C++
  •   C#
  •   GO
  •   Kotlin
  •   Swift
  • 为什么 Go for-range 的 value 值地址每次都一样?


    原文链接: 为什么 Go for-range 的 value 值地址每次都一样?

    循环语句是一种常用的控制结构,在 Go 语言中,除了 for 关键字以外,还有一个 range 关键字,可以使用 for-range 循环迭代数组、切片、字符串、map 和 channel 这些数据类型。

    但是在使用 for-range 循环迭代数组和切片的时候,是很容易出错的,甚至很多老司机一不小心都会在这里翻车。

    具体是怎么翻的呢?我们接着看。

    现象

    先来看两段很有意思的代码:

    无限循环

    如果我们在遍历数组的同时向数组中添加元素,能否得到一个永远都不会停止的循环呢?

    比如下面这段代码:

    func main() {
        arr := []int{1, 2, 3}
        for _, v := range arr {
            arr = append(arr, v)
        }
        fmt.Println(arr)
    }
    

    程序输出:

    $ go run main.go
    1 2 3 1 2 3
    

    上述代码的输出意味着循环只遍历了原始切片中的三个元素,我们在遍历切片时追加的元素并没有增加循环的执行次数,所以循环最终还是停了下来。

    相同地址

    第二个例子是使用 Go 语言经常会犯的一个错误。

    当我们在遍历一个数组时,如果获取 range 返回变量的地址并保存到另一个数组或者哈希时,会遇到令人困惑的现象:

    func main() {
        arr := []int{1, 2, 3}
        newArr := []*int{}
        for _, v := range arr {
            newArr = append(newArr, &v)
        }
        for _, v := range newArr {
            fmt.Println(*v)
        }
    }
    

    程序输出:

    $ go run main.go
    3 3 3
    

    上述代码并没有输出 1 2 3,而是输出 3 3 3。

    正确的做法应该是使用 &arr[i] 替代 &v,像这种编程中的细节是很容易出错的。

    原因

    具体原因也并不复杂,一句话就能解释。

    对于数组、切片或字符串,每次迭代,for-range 语句都会将原始值的副本传递给迭代变量,而非原始值本身。

    口说无凭,具体是不是这样,还得靠源码说话。

    Go 编译器会将 for-range 语句转换成类似 C 语言的三段式循环结构,就像这样:

    // Arrange to do a loop appropriate for the type.  We will produce
    //   for INIT ; COND ; POST {
    //           ITER_INIT
    //           INDEX = INDEX_TEMP
    //           VALUE = VALUE_TEMP // If there is a value
    //           original statements
    //   }
    

    迭代数组时,是这样:

    // The loop we generate:
    //   len_temp := len(range)
    //   range_temp := range
    //   for index_temp = 0; index_temp < len_temp; index_temp++ {
    //           value_temp = range_temp[index_temp]
    //           index = index_temp
    //           value = value_temp
    //           original body
    //   }
    

    切片:

    //   for_temp := range
    //   len_temp := len(for_temp)
    //   for index_temp = 0; index_temp < len_temp; index_temp++ {
    //           value_temp = for_temp[index_temp]
    //           index = index_temp
    //           value = value_temp
    //           original body
    //   }
    

    从上面的代码片段,可以总结两点:

    1. 在循环开始前,会将数组或切片赋值给一个新变量,在赋值过程中就发生了拷贝,迭代的实际上是副本,这也就解释了现象 1。
    2. 在循环过程中,会将迭代元素赋值给一个临时变量,这又发生了拷贝。如果取地址的话,每次都是一样的,都是临时变量的地址。

    以上就是本文的全部内容,如果觉得还不错的话欢迎点赞,转发和关注,感谢支持。


    参考文章:

    • https://garbagecollected.org/2017/02/22/go-range-loop-internals/
    • https://draveness.me/golang/docs/part2-foundation/ch05-keyword/golang-for-range/

    推荐阅读:

    • 为什么 Go 不支持 []T 转换为 []interface
    • 为什么 Go 语言 struct 要使用 tags
  • 相关阅读:
    使用标准信号检测 VM振弦采集模块测量精度修正
    如何配置nginx的转发?
    初识linux(1)
    Slim GAIN(SGAIN)介绍及代码实现——基于生成对抗网络的缺失数据填补
    【QT配置第三方MQTT协议】
    深度学习-通过Resnet18实现CIFAR10数据分类
    FF300R08W2P2B11A 汽车用EasyPACK 模块 2 个独立式
    WPF向Avalonia迁移(四、其他事项)
    Handler的message分为三种
    Docker从入门到上天系列第二篇:传统虚拟机和容器的对比以及Docker的作用以及所解决的问题
  • 原文地址:https://www.cnblogs.com/alwaysbeta/p/17365315.html
  • 最新文章
  • 攻防演习之三天拿下官网站群
    数据安全治理学习——前期安全规划和安全管理体系建设
    企业安全 | 企业内一次钓鱼演练准备过程
    内网渗透测试 | Kerberos协议及其部分攻击手法
    0day的产生 | 不懂代码的"代码审计"
    安装scrcpy-client模块av模块异常,环境问题解决方案
    leetcode hot100【LeetCode 279. 完全平方数】java实现
    OpenWrt下安装Mosquitto
    AnatoMask论文汇总
    【AI日记】24.11.01 LangChain、openai api和github copilot
  • 热门文章
  • 十款代码表白小特效 一个比一个浪漫 赶紧收藏起来吧!!!
    奉劝各位学弟学妹们,该打造你的技术影响力了!
    五年了,我在 CSDN 的两个一百万。
    Java俄罗斯方块,老程序员花了一个周末,连接中学年代!
    面试官都震惊,你这网络基础可以啊!
    你真的会用百度吗?我不信 — 那些不为人知的搜索引擎语法
    心情不好的时候,用 Python 画棵樱花树送给自己吧
    通宵一晚做出来的一款类似CS的第一人称射击游戏Demo!原来做游戏也不是很难,连憨憨学妹都学会了!
    13 万字 C 语言从入门到精通保姆级教程2021 年版
    10行代码集2000张美女图,Python爬虫120例,再上征途
Copyright © 2022 侵权请联系2656653265@qq.com    京ICP备2022015340号-1
正则表达式工具 cron表达式工具 密码生成工具

京公网安备 11010502049817号