为了账号安全,请及时绑定邮箱和手机立即绑定

Go中切片的最大长度

Go中切片的最大长度

Go
12345678_0001 2021-09-10 15:23:54
我在 4Gb 机器的 64 位 linux 操作系统中运行以下代码:package mainimport (    "fmt"    "math")func main() {    r := make([]bool, math.MaxInt32)    fmt.Println("Size: ", len(r))}当我运行这个时,我得到:Size: 2147483647如果我更改math.MaxInt32formath.MaxUint32我得到:fatal error: runtime: out of memory由于math.MaxUint32内存不足,切片大小为我所期望的,但是当我尝试使用时,math.MaxInt64我得到:panic: runtime error: makeslice: len out of range所以显然我不能创建一个大小为 的切片math.MaxInt64,这给我们带来了我的问题:如果内存不是问题,那么我在 Go 中无法创建的最大切片是什么?我记得,在 Java 中,原始数组索引是用 type 管理的int,所以原始数组的最大大小是 an 的最大值int,如果你尝试用long它来做它会引发异常(据我记得) ,和Go一样吗?Go 中的切片索引是否绑定到一种特定类型?编辑:我使用struct{}代替bool和分配math.MaxInt64元素来运行测试。一切都按预期进行,并打印:Size: 9223372036854775807那么,另一个问题,为什么在看起来错误相同(内存不足)时会出现两种不同的错误消息?每个错误弹出的条件是什么?
查看完整描述

1 回答

?
慕仙森

TA贡献1827条经验 获得超7个赞

根据文档,The elements can be addressed by integer indices 0 through len(s)-1. 这意味着切片的最大容量是目标构建上默认整数的大小。


编辑:从查看源代码来看,似乎有一个安全检查来确保这个切片大小是完全可能的:


func makeslice(t *slicetype, len64 int64, cap64 int64) sliceStruct {

    // NOTE: The len > MaxMem/elemsize check here is not strictly necessary,

    // but it produces a 'len out of range' error instead of a 'cap out of range' error

    // when someone does make([]T, bignumber). 'cap out of range' is true too,

    // but since the cap is only being supplied implicitly, saying len is clearer.

    // See issue 4085.

    len := int(len64)

    if len64 < 0 || int64(len) != len64 || t.elem.size > 0 && uintptr(len) > maxmem/uintptr(t.elem.size) {

        panic(errorString("makeslice: len out of range"))

    }

所以在这种情况下,看起来uintptr(len) > maxmem/uintptr(t.elem.size)我们不允许进行这种大小的分配。


但是,当我分配struct{}不占用内存时,允许此大小:


func main(){

    r := make([]struct{}, math.MaxInt64)

    fmt.Println(len(r))

}

// prints 9223372036854775807


查看完整回答
反对 回复 2021-09-10
  • 1 回答
  • 0 关注
  • 258 浏览
慕课专栏
更多

添加回答

举报

0/150
提交
取消
意见反馈 帮助中心 APP下载
官方微信