为什么这段代码会在go(大约81M)中生成非常大的可执行文件?

时间:2013-12-19 00:48:24

标签: go

我是golang的新手。我写了这样的代码:

package main

import (
    "fmt"
)

const (
    BIG_NUM = 10 * 1000 * 1000
)

type BigData [BIG_NUM]uint64

func (self BigData) String() string {
    return fmt.Sprintf("%d\n", self[0])
}

func (self *BigData) Clear() {
    *self = BigData{}
}

func main() {
    data := new(BigData)
    fmt.Println(data)
}

编译后,可执行文件大约是81M:

$ ls -l
-rwxr-xr-x  1 tchen  522017917  81533376 Dec 19 08:44 test
-rw-r--r--  1 tchen  522017917       290 Dec 19 08:44 test.go

当我删除Clear()函数时,可执行文件变为正常大小。那么为什么这个Clear()函数生成如此大的可执行文件?它应该在运行时调用,而不是编译时间,对吗?

------------进一步调查-----------

由@FUZxxl启发,我修改了代码以在Clear()中使用全局变量。这次文件大小恢复正常。因此全局变量正确地放入.bss部分。

package main

import (
    "fmt"
)

const (
    BIG_NUM = 10 * 1000 * 1000
)

type BigData [BIG_NUM]uint64

var (
    bigData = BigData{}
)

func (self BigData) String() string {
    return fmt.Sprintf("%d\n", self[0])
}

func (self *BigData) Clear() {
    *self = bigData
}

func main() {
    data := new(BigData)
    fmt.Println(data)
}

已编译的可执行文件:

$ ls -l
-rwxr-xr-x  1 tchen  522017917  1534384 Dec 19 10:55 test
-rw-r--r--  1 tchen  522017917      318 Dec 19 10:55 test.go

------------最新更新-----------

到目前为止,这似乎是一个编译器问题。已提交错误:https://code.google.com/p/go/issues/detail?id=6993。如果您有兴趣,可以跟进。

在有任何解决方案之前,您应避免在函数中使用x := Y{}。解决方法是(感谢ality@pbrane.org):

func (self *BigData) Clear() {
-   *self = BigData{}
+   var zero BigData
+   *self = zero
} 

1 个答案:

答案 0 :(得分:10)

BigData{}函数中的Clear()看起来导致80,000,000字节的数组= sizeof uint64 * 10 * 1000 * 1000被硬编码到二进制文件中。