当前位置:首页 > 文章列表 > 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互联网时代的弄潮儿。
    511次学习
  • 简单聊聊mysql8与网络通信
    简单聊聊mysql8与网络通信
    如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
    499次学习
  • JavaScript正则表达式基础与实战
    JavaScript正则表达式基础与实战
    在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
    487次学习
  • 从零制作响应式网站—Grid布局
    从零制作响应式网站—Grid布局
    本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
    484次学习
查看更多
AI推荐
  • 千音漫语:智能声音创作助手,AI配音、音视频翻译一站搞定!
    千音漫语
    千音漫语,北京熠声科技倾力打造的智能声音创作助手,提供AI配音、音视频翻译、语音识别、声音克隆等强大功能,助力有声书制作、视频创作、教育培训等领域,官网:https://qianyin123.com
    732次使用
  • MiniWork:智能高效AI工具平台,一站式工作学习效率解决方案
    MiniWork
    MiniWork是一款智能高效的AI工具平台,专为提升工作与学习效率而设计。整合文本处理、图像生成、营销策划及运营管理等多元AI工具,提供精准智能解决方案,让复杂工作简单高效。
    691次使用
  • NoCode (nocode.cn):零代码构建应用、网站、管理系统,降低开发门槛
    NoCode
    NoCode (nocode.cn)是领先的无代码开发平台,通过拖放、AI对话等简单操作,助您快速创建各类应用、网站与管理系统。无需编程知识,轻松实现个人生活、商业经营、企业管理多场景需求,大幅降低开发门槛,高效低成本。
    720次使用
  • 达医智影:阿里巴巴达摩院医疗AI影像早筛平台,CT一扫多筛癌症急慢病
    达医智影
    达医智影,阿里巴巴达摩院医疗AI创新力作。全球率先利用平扫CT实现“一扫多筛”,仅一次CT扫描即可高效识别多种癌症、急症及慢病,为疾病早期发现提供智能、精准的AI影像早筛解决方案。
    737次使用
  • 智慧芽Eureka:更懂技术创新的AI Agent平台,助力研发效率飞跃
    智慧芽Eureka
    智慧芽Eureka,专为技术创新打造的AI Agent平台。深度理解专利、研发、生物医药、材料、科创等复杂场景,通过专家级AI Agent精准执行任务,智能化工作流解放70%生产力,让您专注核心创新。
    714次使用
微信登录更方便
  • 密码登录
  • 注册账号
登录即同意 用户协议隐私政策
返回登录
  • 重置密码