Go语言使用make进行内存分配的代码示例

 更新时间:2025年06月16日 09:14:25   作者:tekin  
在Go语言里,内存分配是程序开发中的关键环节,make是Go语言内置的一个重要函数,专门用于为切片(slice)、映射(map)和通道(channel)这三种引用类型分配内存并初始化,本文将深入剖析 make 函数的使用,结合代码示例与实际项目场景,帮助开发者更好地掌握这一特性

make 函数的基本概念

语法与适用类型

make 函数的语法格式为 make(T, args),其中 T 代表要创建的类型,必须是切片、映射或通道这三种引用类型之一,args 是根据不同类型而定的参数。以下是三种类型使用 make 函数的基本形式:

  • 切片make([]T, length, capacity),其中 T 是切片元素的类型,length 是切片的初始长度,capacity 是切片的初始容量(可省略,默认与长度相同)。
  • 映射make(map[K]V, initialCapacity)K 是键的类型,V 是值的类型,initialCapacity 是映射的初始容量(可省略)。
  • 通道make(chan T, bufferSize)T 是通道中元素的类型,bufferSize 是通道的缓冲区大小(可省略,省略时为无缓冲通道)。

代码示例

package main

import "fmt"

func main() {
    // 使用 make 创建切片
    slice := make([]int, 3, 5)
    fmt.Printf("切片长度: %d, 容量: %d, 内容: %v\n", len(slice), cap(slice), slice)

    // 使用 make 创建映射
    m := make(map[string]int)
    m["apple"] = 1
    m["banana"] = 2
    fmt.Println("映射内容:", m)

    // 使用 make 创建通道
    ch := make(chan int, 2)
    ch <- 10
    ch <- 20
    fmt.Println("从通道接收:", <-ch)
}

在上述代码中,分别使用 make 函数创建了切片、映射和通道,并进行了简单的操作。

切片的 make 分配

长度与容量的区别

在使用 make 创建切片时,长度和容量是两个重要的概念。长度表示切片中当前元素的数量,而容量表示切片底层数组的大小。可以通过 len() 函数获取切片的长度,通过 cap() 函数获取切片的容量。

package main

import "fmt"

func main() {
    // 创建一个长度为 2,容量为 5 的切片
    slice := make([]int, 2, 5)
    fmt.Printf("初始长度: %d, 初始容量: %d\n", len(slice), cap(slice))

    // 向切片追加元素
    slice = append(slice, 1, 2, 3)
    fmt.Printf("追加元素后长度: %d, 容量: %d, 内容: %v\n", len(slice), cap(slice), slice)
}

在这个示例中,初始创建的切片长度为 2,容量为 5。当使用 append 函数追加元素时,如果长度超过了容量,Go 语言会自动重新分配更大的底层数组。

项目场景:数据处理

在数据处理项目中,我们可能需要动态地处理一批数据。使用 make 创建切片可以预先分配一定的容量,减少内存重新分配的次数,提高性能。

package main

import (
    "fmt"
)

func processData() []int {
    // 预先分配容量为 100 的切片
    data := make([]int, 0, 100)
    for i := 0; i < 100; i++ {
        data = append(data, i)
    }
    return data
}

func main() {
    result := processData()
    fmt.Println("处理后的数据:", result)
}

映射的 make 分配

初始容量的作用

在使用 make 创建映射时,指定初始容量可以提高映射的性能。如果预先知道映射可能存储的元素数量,指定合适的初始容量可以减少哈希表扩容的次数。

package main

import "fmt"

func main() {
    // 创建一个初始容量为 10 的映射
    m := make(map[string]int, 10)
    for i := 0; i < 10; i++ {
        key := fmt.Sprintf("key%d", i)
        m[key] = i
    }
    fmt.Println("映射内容:", m)
}

项目场景:缓存系统

在缓存系统中,映射可以用于存储缓存数据。使用 make 创建映射并指定合适的初始容量,可以提高缓存系统的性能。

package main

import (
    "fmt"
)

type Cache struct {
    data map[string]interface{}
}

func NewCache(capacity int) *Cache {
    return &Cache{
        data: make(map[string]interface{}, capacity),
    }
}

func (c *Cache) Set(key string, value interface{}) {
    c.data[key] = value
}

func (c *Cache) Get(key string) (interface{}, bool) {
    val, exists := c.data[key]
    return val, exists
}

func main() {
    cache := NewCache(20)
    cache.Set("item1", 100)
    val, exists := cache.Get("item1")
    if exists {
        fmt.Println("缓存中获取的值:", val)
    }
}

通道的 make 分配

有缓冲通道与无缓冲通道

使用 make 创建通道时,可以指定缓冲区大小。如果不指定缓冲区大小,创建的是无缓冲通道,发送和接收操作会阻塞;如果指定了缓冲区大小,创建的是有缓冲通道,只有当缓冲区满时发送操作才会阻塞,只有当缓冲区为空时接收操作才会阻塞。

package main

import "fmt"

func main() {
    // 创建无缓冲通道
    ch1 := make(chan int)
    go func() {
        ch1 <- 10
        fmt.Println("数据已发送到无缓冲通道")
    }()
    fmt.Println("从无缓冲通道接收:", <-ch1)

    // 创建有缓冲通道
    ch2 := make(chan int, 2)
    ch2 <- 20
    ch2 <- 30
    fmt.Println("从有缓冲通道接收:", <-ch2)
}

项目场景:并发任务处理

在并发任务处理中,通道可以用于协程之间的通信。使用 make 创建合适的通道可以协调不同协程的工作。

package main

import (
    "fmt"
)

func worker(id int, jobs <-chan int, results chan<- int) {
    for j := range jobs {
        fmt.Printf("Worker %d 开始处理任务 %d\n", id, j)
        results <- j * 2
    }
}

func main() {
    const numJobs = 5
    jobs := make(chan int, numJobs)
    results := make(chan int, numJobs)

    // 启动 3 个工作协程
    for w := 1; w <= 3; w++ {
        go worker(w, jobs, results)
    }

    // 发送任务
    for j := 1; j <= numJobs; j++ {
        jobs <- j
    }
    close(jobs)

    // 收集结果
    for a := 1; a <= numJobs; a++ {
        <-results
    }
    close(results)
}

总结

make 函数在 Go 语言中是为切片、映射和通道进行内存分配和初始化的重要工具。通过合理使用 make 函数,可以根据不同的需求为这些引用类型分配合适的内存,提高程序的性能和效率。在实际项目中,无论是数据处理、缓存系统还是并发任务处理,make 函数都发挥着关键作用。开发者需要深入理解 make 函数的使用方法和不同类型的特点,根据具体的场景灵活运用。

以上就是Go语言使用make进行内存分配的代码示例的详细内容,更多关于Go make内存分配的资料请关注脚本之家其它相关文章!

相关文章

  • 详解Go中的高效切片拼接和Go1.22提供的新方法

    详解Go中的高效切片拼接和Go1.22提供的新方法

    在 Go 语言中,切片拼接是一项常见的操作,但如果处理不当,可能会导致性能问题或意外的副作用,本文将详细介绍几种高效的切片拼接方法,希望对大家有所帮助
    2024-01-01
  • Go通过SJSON实现动态修改JSON

    Go通过SJSON实现动态修改JSON

    在Go语言 json 处理领域,在 json 数据处理中,读取与修改是两个核心需求,本文我们就来看看如何使用SJSON进行动态修改JSON吧,有需要的小伙伴可以了解下
    2025-03-03
  • Go中的Timer 和 Ticker详解

    Go中的Timer 和 Ticker详解

    在日常开发中,我们可能会遇到需要延迟执行或周期性地执行一些任务,这个时候就需要用到 Go 语言中的定时器,本文将会对这两种定时器类型进行介绍,感兴趣的朋友一起看看吧
    2024-07-07
  • 详解golang中发送http请求的几种常见情况

    详解golang中发送http请求的几种常见情况

    这篇文章主要介绍了详解golang中发送http请求的几种常见情况,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2019-12-12
  • Go语言如何使用golang-jwt/jwt/v4进行JWT鉴权详解

    Go语言如何使用golang-jwt/jwt/v4进行JWT鉴权详解

    最近项目中需要用到鉴权机制,golang中jwt可以用,这篇文章主要给大家介绍了关于Go语言如何使用golang-jwt/jwt/v4进行JWT鉴权的相关资料,需要的朋友可以参考下
    2022-09-09
  • Go项目在GoLand中导入依赖标红问题的解决方案

    Go项目在GoLand中导入依赖标红问题的解决方案

    这篇文章主要介绍了Go项目在GoLand中导入依赖标红问题的解决方案,文中通过代码示例讲解的非常详细,对大家的学习或工作有一定的帮助,需要的朋友可以参考下
    2024-06-06
  • golang敏感词过滤的实现

    golang敏感词过滤的实现

    本文主要介绍了golang敏感词过滤的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2023-01-01
  • 使用Go http重试请求的示例

    使用Go http重试请求的示例

    开发中对于http请求是经常遇到,一般可能网络延迟或接口返回超时,这篇文章主要介绍了使用Go http重试请求的示例,需要的朋友可以参考下
    2022-08-08
  • Go语言学习之操作MYSQL实现CRUD

    Go语言学习之操作MYSQL实现CRUD

    Go官方提供了database包,database包下有sql/driver。该包用来定义操作数据库的接口,这保证了无论使用哪种数据库,操作方式都是相同的。本文就来和大家聊聊Go语言如何操作MYSQL实现CRUD,希望对大家有所帮助
    2023-02-02
  • Go语言中struct的匿名属性特征实例分析

    Go语言中struct的匿名属性特征实例分析

    这篇文章主要介绍了Go语言中struct的匿名属性特征,实例分析了struct的匿名属性特征,对于深入学习Go语言程序设计具有一定参考借鉴价值,需要的朋友可以参考下
    2015-02-02

最新评论