当前位置:首页 > 文章列表 > Golang > Go教程 > Go语言new和make区别详解

Go语言new和make区别详解

2025-09-02 13:45:38 0浏览 收藏

亲爱的编程学习爱好者,如果你点开了这篇文章,说明你对《Go语言new与make区别解析》很感兴趣。本篇文章就来给大家详细解析一下,主要介绍一下,希望所有认真读完的童鞋们,都有实质性的提高。

深入理解Go语言中的new与make:内存分配与类型初始化

Go语言提供了new和make两种内建函数用于内存分配和初始化,它们各自服务于不同的场景。new用于为任何类型分配零值内存并返回其指针,而make则专为切片、映射和通道这三种引用类型设计,用于分配并初始化其内部数据结构,返回的是已准备好使用的类型实例本身。理解两者的区别对于编写高效且正确的Go代码至关重要。

Go语言的内存分配机制概览

在Go语言中,进行内存分配和值初始化有多种方式,包括:

  • 复合字面量(Composite Literals): 如 Point{2, 3} 或 []int{1, 2, 3},通常在栈上或根据逃逸分析在堆上分配并初始化。
  • 局部变量地址: &someLocalVar,获取已声明局部变量的地址。
  • new 函数: 通用内存分配器,返回指向零值内存的指针。
  • make 函数: 专用于切片、映射和通道的初始化。

理解这些机制有助于我们选择最合适的内存管理方式。

new 关键字:通用内存分配

new 函数是Go语言中一个通用的内存分配器。它的主要功能是:

  1. 分配内存: 为指定类型的值分配足够的内存。
  2. 零值初始化: 将分配的内存初始化为该类型的零值(例如,整型为0,布尔型为false,字符串为空字符串,指针为nil)。
  3. 返回指针: 返回一个指向新分配内存的指针。

语法: new(Type)

示例:

package main

import "fmt"

type Point struct {
    X, Y int
}

func main() {
    // 为Point类型分配内存,并返回*Point类型的指针
    p1 := new(Point)
    fmt.Printf("p1 类型: %T, 值: %+v\n", p1, p1) // p1 类型: *main.Point, 值: &{X:0 Y:0}

    // 为int类型分配内存,并返回*int类型的指针
    i1 := new(int)
    fmt.Printf("i1 类型: %T, 值: %v\n", i1, *i1) // i1 类型: *int, 值: 0

    // 对比复合字面量:&Point{} 结合了分配和初始化
    p2 := &Point{}
    fmt.Printf("p2 类型: %T, 值: %+v\n", p2, p2) // p2 类型: *main.Point, 值: &{X:0 Y:0}

    p3 := &Point{X: 2, Y: 3}
    fmt.Printf("p3 类型: %T, 值: %+v\n", p3, p3) // p3 类型: *main.Point, 值: &{X:2 Y:3}

    // 注意:&int 是非法的,因为int是一个值类型,不能直接取其类型地址。
    // 但 new(int) 是合法的,它分配了一个int的内存并返回其指针。
    // var i int
    // i4 := &i // 合法,获取已存在变量i的地址
}

从示例中可以看出,new(Point) 和 &Point{} 都能得到 *Point 类型的值,但后者允许在分配的同时进行字段初始化。对于基本类型如 int,new(int) 是分配零值 int 并返回其指针的常用方式,因为直接 &int 是语法错误的。

make 关键字:引用类型的专属初始化

make 函数与 new 不同,它不是通用的内存分配器。make 专用于分配并初始化三种内建的引用类型:切片(slice)映射(map)通道(channel)。这三种类型在Go语言中是特殊的存在,它们不仅仅是内存块,还需要内部数据结构(如切片头、哈希表、缓冲区等)进行初始化才能正常使用。

语法:

  • 切片: make([]Type, length, capacity) 或 make([]Type, length)
  • 映射: make(map[KeyType]ValueType, initialCapacity) 或 make(map[KeyType]ValueType)
  • 通道: make(chan Type, bufferCapacity) 或 make(chan Type)

make 函数会返回一个已初始化且可用的类型实例本身,而不是指针。

示例:

package main

import "fmt"

func main() {
    // 切片:分配一个长度为5,容量为10的int切片
    s := make([]int, 5, 10)
    fmt.Printf("s 类型: %T, 值: %v, 长度: %d, 容量: %d\n", s, s, len(s), cap(s))
    // s 类型: []int, 值: [0 0 0 0 0], 长度: 5, 容量: 10

    // 映射:分配一个string到int的映射,并初始化其内部哈希表
    m := make(map[string]int)
    fmt.Printf("m 类型: %T, 值: %v\n", m, m)
    // m 类型: map[string]int, 值: map[]
    m["key"] = 10
    fmt.Println("m['key']:", m["key"])

    // 通道:分配一个int类型的通道,带有一个缓冲区
    c := make(chan int, 1)
    fmt.Printf("c 类型: %T, 值: %v\n", c, c)
    // c 类型: chan int, 值: 0xc000060060 (通道的内部表示)
    c <- 1
    val := <-c
    fmt.Println("从通道接收到的值:", val)

    // 注意:make(Point) 或 make(int) 是非法的,因为make不能用于非引用类型
    // make(Point) // 编译错误
    // make(int)   // 编译错误
}

可以看到,make 返回的是 []int、map[string]int、chan int 这些类型本身,而不是它们的指针。这是因为这些类型在使用前需要进行特定的初始化步骤,而make正是负责完成这些步骤。

new 与 make 的关键区别

理解 new 和 make 的核心差异是掌握Go内存管理的关键:

  1. 返回类型:

    • new(T) 返回 *T,即一个指向零值 T 的指针。
    • make(T, args) 返回 T,即一个已初始化且可用的 T 类型实例。
      p := new(chan int)   // p 的类型是 *chan int
      c := make(chan int)  // c 的类型是 chan int

      这里 p 是一个指向 nil 通道的指针,需要进一步解引用并赋值一个 make 创建的通道才能使用。而 c 已经是可以直接使用的通道。

  2. 适用类型:

    • new 可以用于任何类型(包括结构体、基本类型、切片、映射、通道等),它只是分配内存并零值化。
    • make 只能用于切片 ([]T)、映射 (map[K]V) 和通道 (chan T) 这三种引用类型。
  3. 初始化行为:

    • new 只是将内存清零,使其达到该类型的零值状态。对于引用类型,如 new([]int),它会返回一个指向 nil 切片头的指针。这个切片仍然是 nil,不能直接使用(例如,不能 append)。
    • make 不仅分配内存,还会初始化这些引用类型的内部数据结构,使它们处于可用状态。例如,make([]int, 0, 5) 会创建一个长度为0,容量为5的切片头,并指向一个底层的数组。

设计考量:为何需要两个函数?

Go语言的设计者选择保留 new 和 make 两个独立的函数,而非合并为一个,主要是出于清晰性避免混淆的考虑。

试想如果只有一个名为 NEW 的函数:

  • NEW(*int) 对应 new(int)
  • NEW(*Point) 对应 new(Point)
  • NEW(*chan int) 对应 new(chan int) (返回 *chan int)
  • NEW(chan int) 对应 make(chan int) (返回 chan int)
  • NEW([]int, 10) 对应 make([]int, 10)

这种统一的 NEW 函数在处理引用类型时,需要根据参数是否带有 * 来区分是分配指针还是初始化实例,这无疑会增加学习曲线和使用时的心智负担。例如,NEW(chan int) 返回 chan int,而 NEW(*chan int) 返回 *chan int,这种细微的语法差异可能会导致开发者混淆。

通过将功能明确地划分为 new(通用零值内存分配,返回指针)和 make(引用类型初始化,返回实例),Go语言使得这两种操作的目的和结果更加直观,降低了新Go程序员的理解难度。

总结与实践建议

  • 使用 new:

    • 当你需要为任何类型分配内存并获取一个指向其零值的指针时。
    • 当你想明确地表示你只是在分配内存,而初始化将在后续步骤中进行时。
    • 例如:ptr := new(MyStruct),counter := new(int)。
  • 使用 make:

    • 当你需要创建和初始化切片、映射或通道时。
    • 这些类型在创建时需要特定的内部结构设置,make 确保它们在返回时是完全可用的。
    • 例如:s := make([]int, 10), m := make(map[string]string), ch := make(chan int, 5)。
  • 结构体初始化: 对于结构体,通常推荐使用复合字面量 &MyStruct{Field: value} 的形式,它结合了分配和初始化,代码更简洁易读。

理解 new 和 make 的区别,并根据具体场景选择合适的函数,是编写高效、健壮Go程序的关键一步。

以上就是本文的全部内容了,是否有顺利帮助你解决问题?若是能给你带来学习上的帮助,请大家多多支持golang学习网!更多关于Golang的相关知识,也可关注golang学习网公众号。

Golang包冲突解决与别名使用方法Golang包冲突解决与别名使用方法
上一篇
Golang包冲突解决与别名使用方法
使用filter\_input实现自定义验证方法
下一篇
使用filter\_input实现自定义验证方法
查看更多
最新文章
查看更多
课程推荐
  • 前端进阶之JavaScript设计模式
    前端进阶之JavaScript设计模式
    设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
    543次学习
  • GO语言核心编程课程
    GO语言核心编程课程
    本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
    516次学习
  • 简单聊聊mysql8与网络通信
    简单聊聊mysql8与网络通信
    如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
    500次学习
  • JavaScript正则表达式基础与实战
    JavaScript正则表达式基础与实战
    在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
    487次学习
  • 从零制作响应式网站—Grid布局
    从零制作响应式网站—Grid布局
    本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
    485次学习
查看更多
AI推荐
  • ChatExcel酷表:告别Excel难题,北大团队AI助手助您轻松处理数据
    ChatExcel酷表
    ChatExcel酷表是由北京大学团队打造的Excel聊天机器人,用自然语言操控表格,简化数据处理,告别繁琐操作,提升工作效率!适用于学生、上班族及政府人员。
    3193次使用
  • Any绘本:开源免费AI绘本创作工具深度解析
    Any绘本
    探索Any绘本(anypicturebook.com/zh),一款开源免费的AI绘本创作工具,基于Google Gemini与Flux AI模型,让您轻松创作个性化绘本。适用于家庭、教育、创作等多种场景,零门槛,高自由度,技术透明,本地可控。
    3405次使用
  • 可赞AI:AI驱动办公可视化智能工具,一键高效生成文档图表脑图
    可赞AI
    可赞AI,AI驱动的办公可视化智能工具,助您轻松实现文本与可视化元素高效转化。无论是智能文档生成、多格式文本解析,还是一键生成专业图表、脑图、知识卡片,可赞AI都能让信息处理更清晰高效。覆盖数据汇报、会议纪要、内容营销等全场景,大幅提升办公效率,降低专业门槛,是您提升工作效率的得力助手。
    3436次使用
  • 星月写作:AI网文创作神器,助力爆款小说速成
    星月写作
    星月写作是国内首款聚焦中文网络小说创作的AI辅助工具,解决网文作者从构思到变现的全流程痛点。AI扫榜、专属模板、全链路适配,助力新人快速上手,资深作者效率倍增。
    4543次使用
  • MagicLight.ai:叙事驱动AI动画视频创作平台 | 高效生成专业级故事动画
    MagicLight
    MagicLight.ai是全球首款叙事驱动型AI动画视频创作平台,专注于解决从故事想法到完整动画的全流程痛点。它通过自研AI模型,保障角色、风格、场景高度一致性,让零动画经验者也能高效产出专业级叙事内容。广泛适用于独立创作者、动画工作室、教育机构及企业营销,助您轻松实现创意落地与商业化。
    3814次使用
微信登录更方便
  • 密码登录
  • 注册账号
登录即同意 用户协议隐私政策
返回登录
  • 重置密码