2

我正在尝试使用 Go 来KiB, MiB, ..., ZiB, Yib分别获取KibiByte, MebiByte, ..., ZebiByte, YobiByte.

我在 Golang 中的代码是:

package main 
import ( 
    "fmt"
)

func main() {
    s := []string{"KiB", "MiB", "GiB", "TiB", "PiB", "EiB", "ZiB", "YiB"}

    for k,v := range(s) {
        fmt.Printf("%s: %v\n", v, 1 << uint64(10 * (k+1)))
    }
}

但是,ZiB and YiB溢出 Go的值,uint64这就是为什么我有这个输出:

KiB: 1024
MiB: 1048576
GiB: 1073741824
TiB: 1099511627776         // exceeds 1 << 32
PiB: 1125899906842624
EiB: 1152921504606846976
ZiB: 0                    // exceeds 1 << 64
YiB: 0                    // exceeds 1 << 64

否则,Python3在此代码中使用相同的移位逻辑:

a = ["KiB", "MiB", "GiB", "TiB", "PiB", "EiB", "ZiB", "YiB"]
for k,v in enumerate(a):
    print("{}: {}".format(v, 1 << (10 *(k+1))))

输出是正确的,如下面的输出:

KiB: 1024
MiB: 1048576
GiB: 1073741824
TiB: 1099511627776
PiB: 1125899906842624
EiB: 1152921504606846976
ZiB: 1180591620717411303424
YiB: 1208925819614629174706176

那么,我怎样才能绕过 Gouint64限制并使用移位整数获得正确的值,就像我可以从使用 Python 移位整数中得到的一样。

谢谢。

4

1 回答 1

9

您不能使用原始 uint64 处理需要超过 64 位的数字。Python 具有任意精度的整数,要在 Go 中获得相同的精度,您需要使用该math/big包。

s := []string{"KiB", "MiB", "GiB", "TiB", "PiB", "EiB", "ZiB", "YiB"}

one := big.NewInt(1)
for k, v := range s {
    fmt.Printf("%s: %v\n", v, new(big.Int).Lsh(one, uint(10*(k+1))))
}

https://play.golang.org/p/i5v5P5QgQb

于 2017-02-09T18:33:12.503 回答