go 语言中 append 的问题

在工作中写了如下一段代码:

  for _, tag := range tags {
    args := append(fgtArgs, "--tag", tag)
    cmd := utils.Command{
      ...
      Arguments: args,
    }
    cmds = append(cmds, cmd)
  }

虽然 tags 中的各个 tag 不相同,但最后,所有的 cmd 中都使用了最后一个 tag。
经跟踪发现,在循环体中 args 是变化的,但在变化之后就覆盖了以前的 args,它们之间并没有保持独立。即 args 看似是局部变量,实际上跟全局变量类似。经测试,发现原因出在 fgtArgs 是由 append 生成的
看如下测试代码:

  a := append([]string{"abc", "abc", "abc", "abc", "abc", "abc", "abc", "abc", "abc", "abc", "abc", "abc", "abc", "abc", "abc", "abc"}, "000")
  // a := []string{"abc", "abc", "abc", "abc", "abc", "abc", "abc", "abc", "abc", "abc", "abc", "abc", "abc", "abc", "abc", "abc", "000"}
  // a := append([]string{"abc"}, "000")
  b := append(a, "123")
  c := append(a, "456")

  fmt.Println(a)
  fmt.Println(b)
  fmt.Println(c)

当采用 append 初始化 a 时(第1行),最后运行结果中 b 和 c 的值是一样的,最后都是“456”,当采用数组初始化的方式初始化 a 时(第2行),最后的运行结果中 b 和 c 的值是不同的。
但是,若用少量元素初始化 a,即使使用 append(第3行),b 和 c 的值也是不同的。
这应该是跟 append 为数组开辟空间及go数组的实现方式有关,没有深入研究。

最终的解决办法就是每次创建新的拷贝,而不是在原来的 fgtArgs 上生成 args,代码如下:

  for _, tag := range tags {
    args := append(fgtArgs[:0:0], fgtArgs...)
    args = append(fgtArgs, "--tag", tag)
    cmd := utils.Command{
      ...
      Arguments: args,
    }
    cmds = append(cmds, cmd)
  }
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 以下内容是我在学习和研究Go时,对Go的特性、重点和注意事项的提取、精练和总结,还有一些学习笔记(注:部分笔记是摘...
    科研者阅读 3,772评论 0 1
  • 1. 摘要 本文是慕课网上郝林的《Go语言第一课》的学习笔记。作为一名老码农,最近才下定决心来学习新的语言,有点惭...
    笔名辉哥阅读 5,716评论 0 51
  • 1.相对路径和绝对路径的区别: 绝对路径是指从根目录开始的完整路径 相对路径是相对于当前工作目录的路径,不是唯一的...
    三月李壮阅读 864评论 0 2
  • Go入门 Go介绍 部落图鉴之Go:爹好还这么努力? 环境配置 安装 下载源码编译安装 下载相应平台的安装包安装 ...
    齐天大圣李圣杰阅读 10,086评论 0 26
  • 定义变量 Go语言里面定义变量有很多种方式。使用var关键字是Go最基本的定义变量方式,与C语言不同的是Go把变量...
    87d6dc4b11a7阅读 2,540评论 0 4