Go 语言是如何实现切片扩容

简介: Go 语言是如何实现切片扩容

Go 语言中的切片(slice)是一个非常灵活的数据结构,它可以根据需求自动扩容。slice 的扩容机制如下:

  1. 初始容量:

    • 当使用 make([]T, len, cap) 创建 slice 时,可以指定初始长度 len 和容量 cap
    • 如果只指定长度,则容量默认等于长度。
  2. 添加元素时的扩容:

    • 当向 slice 中添加元素,并且当前容量不足时,Go 运行时会自动扩容。
    • 扩容策略是: 新容量 = 旧容量 + max(旧容量, 1)。
    • 也就是说,容量每次会至少翻倍,直到满足新增元素的需求。
  3. 扩容过程:

    • 分配一块新的连续内存空间,容量是旧容量的 1.5 倍或更大。
    • 将旧 slice 中的元素拷贝到新内存空间。
    • 更新 slice 的底层数组指针、长度和容量。
  4. 复制数据:

    • 扩容时会将旧 slice 中的数据完整地拷贝到新的内存空间。
    • 这意味着如果 slice 中的元素是指针或其他引用类型,那么这些引用也会被拷贝。

示例代码:

package main

import "fmt"

func main() {
   
    // 初始容量为 3
    s := make([]int, 0, 3)

    // 添加元素直到触发扩容
    for i := 0; i < 10; i++ {
   
        s = append(s, i)
        fmt.Printf("len=%d, cap=%d\n", len(s), cap(s))
    }
}

输出:

len=1, cap=3
len=2, cap=3
len=3, cap=3
len=4, cap=6
len=5, cap=6
len=6, cap=6
len=7, cap=9
len=8, cap=9
len=9, cap=9
len=10, cap=15

从输出中可以看到,slice 的容量每次都会增加一倍以上,直到满足新增元素的需求。这种扩容机制保证了 slice 的灵活性和性能。

相关文章
|
1天前
|
编译器 Go 开发者
Go语言语法基础入门
Go语言语法基础入门
4 0
|
1天前
|
Devops Go 云计算
Go语言发展现状:历史、应用、优势与挑战
Go语言发展现状:历史、应用、优势与挑战
9 2
|
2天前
|
Go
go切片和map比较
go切片和map比较
5 0
|
2天前
|
Go
go语言map、实现set
go语言map、实现set
9 0
|
2天前
|
Go
go语言数组与切片
go语言数组与切片
10 0
|
2天前
|
Go
go语言的hello,world
go语言的hello,world
7 1
|
2天前
|
Java Go Windows
go语言实现加减法出题器(再也不用担心孩子学习了)
go语言实现加减法出题器(再也不用担心孩子学习了)
6 0
|
4天前
|
存储 中间件 Go
在go语言服务中封装路由和示例
【6月更文挑战第23天】本文介绍golang后端按协议处理、中间件(一次性与每次请求执行)划分、以及服务架构Controller、Logic/Service、DAO/Repository和Routers划分。代码仓库在GitHub上提供。使用框架简化了交互和处理。后续章节深入探讨服务构建。
104 5
在go语言服务中封装路由和示例
|
5天前
|
安全 测试技术 Go
Go语言在高并发场景下的应用
在当今互联网高速发展的时代,高并发已成为众多应用系统面临的核心问题。本文探讨了Go语言在高并发场景下的优势,并通过具体实例展示了其在实际应用中的效果和性能表现。
|
5天前
|
Unix Go 开发者
探索Go语言并发模型:原理与实践
本文深入探讨了Go语言的并发模型,包括其设计原理、Goroutine和Channel的基本使用,以及如何在实际项目中高效地应用这些概念来提高程序的性能和可维护性。