我的编程空间,编程开发者的网络收藏夹
学习永远不晚

Golang中for循环遍历避坑指南

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

Golang中for循环遍历避坑指南

一、for循环

循环:让程序多次执行相同的代码块

for循环是Go语言中唯一一个循环结构

for循环经典语法

  • 先执行表达式1
  • 执行表达式2判断是否成立,如果成立执行循环体
  • 循环体执行完成后,执行表达式3
  • 再次执行表达式2,判断是否成立.

for循环用的最多的地方就是遍历数组或切片等

for 表达式1;表达式2;表达式3{
  //循环体
} 

经典for循环结构中 , for关键字后面有三个表达式,且每个表达式都可以省略。

for i := 0; i < 5; i++ {
    fmt.Println(i)
}
//等价于
j := 0
for ; j < 5; {
    fmt.Println(j)
    j++
}

for关键字后面也可以只有一个表达式,表示如果条件成立执行循环体代码。

for i := 0; i < 5; i++ {
    fmt.Println(i)
}
//等价于
j := 0
for j < 5 {
    fmt.Println(j)
    j++
}

二、range语句

Golang range类似迭代器操作,可以对 slice、map、数组、字符串等进行迭代循环。在字符串、数组和切片中它返回 (索引, 值) ,在map中返回 (键, 值),但若当只有一个返回值时,第一个参数是索引或键。

str := "abc"
for i, char := range str {
    fmt.Printf("%d => %s\n", i, string(char))
}
for i := range str { //只有一个返回值
    fmt.Printf("%d\n", i)
}
nums := []int{1, 2, 3}
for i, num := range nums {
    fmt.Printf("%d => %d\n", i, num)
}
kvs := map[string]string{"a": "apple", "b": "banana"}
for k, v := range kvs {
    fmt.Printf("%s => %s\n", k, v)
}
for k := range kvs { //只有一个返回值
    fmt.Printf("%s\n", k)
}
// 输出结果
// 0 => a
// 1 => b
// 2 => c
// 0
// 1
// 2
// 0 => 1
// 1 => 2
// 2 => 3
// a => apple
// b => banana
// a
// b

for循环尤其是range语句,在平时开发过程中频繁使用,但很多开发者经常会在以下场景中踩坑。

场景一,使用循环迭代器的变量

先来看一个明显的错误:

func main() {
    var out []*int
    for i := 0; i < 3; i++ {
        // i := i
        out = append(out, &i)
    }
    fmt.Println("值:", *out[0], *out[1], *out[2])
    fmt.Println("地址:", out[0], out[1], out[2])
}
// 输出结果
// 值: 3 3 3
// 地址: 0xc000012090 0xc000012090 0xc000012090

分析

out是一个整型指针数组变量,在for循环中,声明了一个i变量,每次循环将i的地址追加到out切片中,但是每次追加的其实都是i变量,因此我们追加的是一个相同的地址,而该地址最终的值是3。

正确做法

解开代码中的注释// i := i,每次循环时都重新创建一个新的i变量。

注意,for循环表达式1定义的变量i相当于循环体里面的全局变量,循环体里面定义的变量i相当于循环体里面的局部变量,由于在代码块中同名局部变量会覆盖同名的全局变量,所以此时在循环体逻辑里面,使用的i变量是循环体里面定义的局部变量i。

再看一个比较隐秘的错误:

func main() {
    a1 := []int{1, 2, 3}
    a2 := make([]*int, len(a1))
    for i, v := range a1 {
        a2[i] = &v
    }
    fmt.Println("值:", *a2[0], *a2[1], *a2[2])
    fmt.Println("地址:", a2[0], a2[1], a2[2])
}
// 输出结果
// 值: 3 3 3
// 地址: 0xc000012090 0xc000012090 0xc000012090

分析

大多数人就是在range这里给变量赋值的时候踩坑,因为比较隐秘,其实情况和上面的一样,range在遍历值类型时,其中的v是一个局部变量,只会声明初始化一次,之后每次循环时重新赋值覆盖前面的,所以给a2[i]赋值的时候其实都是同一个地址&v,而v最终的值为a1最后一个元素的值,也就是3。

正确做法

①a2[i]赋值时传递原始指针,即a2[i] = &a1[i]

②创建临时变量t := v;a2[i] = &t

③闭包(与②原理一样),func(v int) { a2[i] = &v }(v)

更为隐秘的还有:

func main() {
    var out [][]int
    for _, i := range [][1]int{{1}, {2}, {3}} {
        out = append(out, i[:])
    }
    fmt.Println("Values:", out)
}
// 输出结果
// [[3] [3] [3]]

原理也是一样的,不论遍历多少次,i[:]总是被本次遍历的值所覆盖

场景二,在循环体内使用goroutines

func main() {
    values := []int{1, 2, 3}
    wg := sync.WaitGroup{}
    for _, val := range values {
        wg.Add(1)
        go func() {
            fmt.Println(val)
            wg.Done()
        }()
    }
    wg.Wait()
}
// 输出结果
// 3
// 3
// 3

分析

对于主协程来讲,循环是很快就跑完的,而这个时候各个协程可能才开始跑,此时val的值已经遍历到最后一个了,所以各协程都输出了3。(如果遍历数据庞大,主协程遍历耗时较久的话,goroutine的输出会根据当时候的val的值,所以每次的输出结果不一定相同的。)

解决办法

①使用临时变量

for _, val := range values {
    wg.Add(1)
    val := val
    go func() {
        fmt.Println(val)
        wg.Done()
    }()
}

②使用闭包

for _, val := range values {
    wg.Add(1)
    go func(val int) {
        fmt.Println(val)
        wg.Done()
    }(val)
}

三、实战

 基于第一、第二小节内容,接下来进行一个Golang for循环实战,代码详情请看注释。

package main
 
import "fmt"
 
func main() {
    slic := []int{1, 2, 3}
    // len(cp) = 3, cap(cp) = 3
    cp := make([]*int, len(slic))
    for i, x := range slic {
        cp[i] = &slic[i]
        cp = append(cp, &x)
    }
 
    // 第一次循环
    // cp[0] = &slic[0]  *cp[0] = 1
    // 由于len=cap=3,append方法的作用是将元素追加到切片的末尾,即cp(len)位置,此时append便超过切片容量,切片进行扩容,将容量扩容2倍(切片容器<1024时,扩容*2)。
    // append后相当于 cp[3] = &x  此时x的值为1,len(cp)=4, cap(cp)=6
 
    // 第二次循环
    // cp[1] = &slic[1]  *cp[1] = 2
    // 再次append相当于 cp[4] = &x 此时x的值为2,len(cp)=5, cap(cp)=6
 
    // 第三次循环
    // cp[2] = &slic[2]  *cp[2] = 3
    // 再次append相当于 cp[5] = &x 此时x的值为3,len(cp)=6, cap(cp)=6
 
    // 至此循环完毕,cp中数据为  &slic[0]  &slic[1] &slic[2] &x  &x  &x
 
      println("len==", len(cp), "  cap==", cap(cp))
      for _, x := range cp {
        fmt.Print(*x, " ")
      }
}
 

到此这篇关于Golang中for循环遍历避坑指南的文章就介绍到这了,更多相关Golang for循环内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

Golang中for循环遍历避坑指南

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

Golang中for循环遍历避坑指南

这篇文章主要为大家详细介绍了Golang中for循环遍历会出现的一些小坑以及对应的解决办法,文中的示例代码讲解详细,感兴趣的可以了解一下
2023-05-20

php中for循环能遍历数组吗

这篇文章主要介绍了php中for循环能遍历数组吗的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇php中for循环能遍历数组吗文章都会有所收获,下面我们一起来看看吧。在php中,for循环能遍历数组,语法为“fo
2023-07-04

php中for循环能不能遍历数组

在php中,for循环能遍历数组,语法为“for($i=0;$i<count($array);$i++){ //处理数组元素的语句块;} ”。使用for遍历数组的限制:1、所遍历的数组必须是索引数组(即下标为数字的数组),不能是关联数组(下标为字符串的数组);2、索引数组的下标必须是连续的整数。
2022-11-17

python中怎么使用for循环遍历列表

在Python中,通过for循环可以轻松遍历列表中的每个元素。例如,假设有一个列表numbers,其中包含一组数字,可以使用for循环来遍历这个列表:numbers = [1, 2, 3, 4, 5]for number in numbe
python中怎么使用for循环遍历列表
2024-04-08

【笔记】Java中for循环遍历删除操作

在Java中,有些场景需要遍历集合中的元素,然后根据条件进行删除元素的操作。如果使用传统的for循环遍历方式来删除元素,很可能出错或发生意想不到的问题。推荐使用迭代器iterator完成 1. 推荐使用迭代器方式****删除 阿里
2023-08-24

在golang中如何循环遍历字符串接口

php小编柚子为您带来关于在golang中如何循环遍历字符串接口的简洁解答。在golang中,字符串是以字节的形式存储的,因此要遍历字符串,我们需要先将其转换为字节数组。通过使用range关键字,我们可以轻松地遍历字符串,并获取每个字节的索
在golang中如何循环遍历字符串接口
2024-02-09

python中重启for循环,使其重新开始遍历

这篇文章主要介绍了python中重启for循环,使其重新开始遍历方式,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
2023-03-02

怎么在Shell脚本中使用for循环遍历参数

这篇文章将为大家详细讲解有关怎么在Shell脚本中使用for循环遍历参数,文章内容质量较高,因此小编分享给大家做个参考,希望大家阅读完这篇文章后对相关知识有一定的了解。1.当一个脚本需要传入的参数较多时,可以使用for循环进行参数遍历示例:
2023-06-09

linux shell 中数组的定义和for循环遍历的方法

linux shell中的语法和普通编程语言 c/c++ java ULmITgaog的不太一样,平时用的不多,所以总是记不住,写脚本才会去查怎么用。 今天突然被问到数组怎么去遍历。平时写shell脚本也经常遍历数组,但是一下没答上
2022-06-04

Java中的for-each循环与迭代器遍历有何区别?(Java中的for-each循环与迭代器遍历方式之间有哪些不同?)

Java中的for-each循环和迭代器遍历是遍历集合的两种方式。for-each循环简洁易用,但缺乏灵活性且不能并发遍历。迭代器遍历提供了更多的控制和灵活性,但性能可能较低,需要更多的代码。选择哪种方法取决于具体需求:需要快速、简洁的遍历且不需要修改/删除元素:for-each循环需要修改/删除元素、反向遍历或并发遍历:迭代器遍历
Java中的for-each循环与迭代器遍历有何区别?(Java中的for-each循环与迭代器遍历方式之间有哪些不同?)
2024-04-02

python中怎么重启for循环使其重新开始遍历

这篇“python中怎么重启for循环使其重新开始遍历”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“python中怎么重启f
2023-07-05

Java中如何使用for循环遍历数组并处理元素?(在Java中,如何通过for循环来遍历数组并对每个元素进行操作?)

本篇文章介绍了如何在Java中使用for循环遍历数组并处理各个元素。通过for循环,可以设置循环变量,检查循环条件,并递增循环变量以遍历整个数组。可以使用标准for循环或增强型for循环(也称为for-each循环),后者提供了一种更简洁的遍历方式。通过遍历数组,可以打印、修改或执行其他操作,例如将每个元素乘以特定值。
Java中如何使用for循环遍历数组并处理元素?(在Java中,如何通过for循环来遍历数组并对每个元素进行操作?)
2024-04-02

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录