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 的灵活性和性能。

相关文章
|
6天前
|
Go
Go 语言循环语句
在不少实际问题中有许多具有规律性的重复操作,因此在程序中就需要重复执行某些语句。
14 1
|
5天前
|
Go 开发者
探索Go语言的并发之美
在Go语言的世界里,"并发"不仅仅是一个特性,它是一种哲学。本文将带你领略Go语言中goroutine和channel的魔力,揭示如何通过Go的并发机制来构建高效、可靠的系统。我们将通过一个简单的示例,展示如何利用Go的并发特性来解决实际问题,让你的程序像Go一样,轻盈而强大。
|
6天前
|
JSON Go API
使用Go语言和Gin框架构建RESTful API:GET与POST请求示例
使用Go语言和Gin框架构建RESTful API:GET与POST请求示例
|
6天前
|
Go
go语言创建字典
go语言创建字典
|
7天前
|
安全 Go 数据处理
探索Go语言的并发之美:Goroutines与Channels
在Go语言的世界里,"并发"不仅仅是一个概念,它是一种生活的方式。本文将带你领略Go语言中Goroutines和Channels的魔力,它们是如何让并发编程变得既简单又高效。我们将通过一个简单的示例,展示如何使用这些工具来构建一个高性能的网络服务。
|
4天前
|
Go 索引
Go to Learn Go之切片
Go to Learn Go之切片
11 1
|
7天前
|
关系型数据库 Go 数据处理
高效数据迁移:使用Go语言优化ETL流程
在本文中,我们将探索Go语言在处理大规模数据迁移任务中的独特优势,以及如何通过Go语言的并发特性来优化数据提取、转换和加载(ETL)流程。不同于其他摘要,本文不仅展示了Go语言在ETL过程中的应用,还提供了实用的代码示例和性能对比分析。
|
7天前
|
Go 定位技术 索引
Go 语言Map(集合) | 19
Go 语言Map(集合) | 19
|
7天前
|
Go
go语言注释,标识符 | 17
go语言注释,标识符 | 17
|
6天前
|
NoSQL Go API
go语言操作Redis
go语言操作Redis