Golang的make和new函数有什么区别 对比内存分配方式的底层差异

new用于分配任意类型的内存并返回指向零值的指针,而make专用于初始化切片、映射和通道并返回已初始化实例。1.new(t)为类型t分配清零内存并返回*t指针,适用于基本类型、结构体等;2.make仅用于创建切片、映射和通道,会初始化其内部结构使其可直接使用;3.声明变量时零值可能为nil(如切片、映射、通道),需make确保可用性;4.new是通用内存分配器,make则是特定复合类型的构造函数,封装了复杂初始化逻辑。

Golang的make和new函数有什么区别 对比内存分配方式的底层差异

Golang中的

make

new

函数,它们的核心区别在于用途和返回值的类型。简单来说,

new

负责分配内存并返回一个指向零值(zeroed value)的指针,而

make

则专门用于初始化切片(slice)、映射(map)和通道(channel)这三种内建类型,并返回一个已初始化(非零值)的实例。

new

是通用的内存分配器,而

make

则是特定复合数据结构的初始化器。

Golang的make和new函数有什么区别 对比内存分配方式的底层差异

解决方案

new

函数的工作方式相对直观。当你调用

new(T)

时,它会为类型

T

分配一块内存,并将这块内存清零(即所有位都设为零),然后返回一个指向这块内存的

*T

类型指针。这里的“零值”指的是该类型在Go语言中的默认初始值,比如整型是0,布尔型是false,字符串是空字符串””,指针是nil。

new

可以用于任何类型,无论是基本类型、结构体还是数组,它只是单纯地分配并清零内存,不进行任何额外的初始化操作。

make

函数则不同,它并非一个通用的内存分配函数。它仅限于创建切片、映射和通道。

make

的职责不仅仅是分配内存,更重要的是它会初始化这些复合数据结构的内部状态,使它们能够立即被使用。例如:

立即学习“go语言免费学习笔记(深入)”;

Golang的make和new函数有什么区别 对比内存分配方式的底层差异切片 (Slice):

make([]T, length, capacity)

会分配一个底层数组,并创建一个切片头(包含指向底层数组的指针、长度和容量)来引用这个数组。这个底层数组会被零值化,但切片头本身是根据你提供的

length

capacity

参数进行初始化的,而不是零值。映射 (Map):

make(map[K]V)

会分配一个哈希表结构,并初始化其内部状态,使其能够存储键值对。这包括分配桶(buckets)、设置负载因子等,而不是简单地清零一块内存。通道 (Channel):

make(chan T, bufferSize)

会分配一个环形缓冲区以及相关的互斥锁和等待队列,并初始化这些结构,使其能够进行发送和接收操作。

因此,

new

是“给我一块干净的内存,我来决定怎么用”,而

make

则是“给我一个能直接用的切片/映射/通道,你帮我把它的内部结构都搭好”。

Golang中何时应该使用make,何时使用new?

选择

make

还是

new

,很大程度上取决于你想要创建的类型以及你希望如何使用它。这并不是一个非此即彼的难题,更多的是关于理解Go语言对不同数据结构的设计哲学。

Golang的make和new函数有什么区别 对比内存分配方式的底层差异

当你需要为任何类型(包括自定义结构体、基本类型如

int

string

等)分配内存,并希望得到一个指向该类型零值的指针时,

new

是你的选择。比如,如果你想创建一个结构体的实例,并且希望通过指针来操作它,或者这个结构体比较大,你不想在栈上分配它(尽管Go的逃逸分析会帮你决定),那么

new(MyStruct)

就非常合适。它给你一个

*MyStruct

,所有字段都是其默认的零值。如果你只是声明一个变量,

var myStruct MyStruct

也会得到一个零值化的

MyStruct

,但它是一个值类型,而不是指针。

然而,对于切片、映射和通道这三种类型,你几乎总是需要使用

make

来初始化它们,除非你明确知道你想要一个

nil

的切片、映射或通道。声明一个

var mySlice []int

会得到一个

nil

切片,对其进行append操作通常没问题(Go运行时会处理),但如果你想直接通过索引访问或对

map

进行读写,

nil

会导致运行时错误(panic)。

make

确保这些复合类型被正确地构造,拥有必要的内部数据结构和容量,从而能够立即进行操作。例如,

make([]int, 5)

会创建一个长度为5的切片,底层有5个零值化的整数;

make(map[string]int)

会创建一个空的、可用的映射。我个人觉得,对于这三者,记住“能用”是关键,

make

就是为了让它们“能用”。

深入理解Golang中make和new的内存分配机制差异

从内存分配的底层视角来看,

new

make

的行为差异更为显著。

new

函数执行的是一个相对简单的内存分配过程。当你调用

new(T)

时,Go运行时会在堆上(Heap)分配一块足够存储类型

T

值的内存空间。这块内存随后会被全部置为零。这个过程就像是操作系统给你一块干净的、空白的土地,你可以随意在上面建造你的房子(数据结构)。因为Go有垃圾回收机制,你不需要手动管理这块内存的释放。它的主要特点是通用性和“零值化”。无论你分配的是一个简单的

int

,还是一个复杂的嵌套结构体,

new

都只是分配并清零相应的字节数。

make

则不仅仅是分配内存那么简单,它还包含了类型特定的初始化逻辑。

切片 (Slice): 当你

make([]T, len, cap)

时,Go运行时会在堆上分配一个底层数组(大小由

cap

决定)。这个数组的元素会被初始化为

T

的零值。同时,

make

还会创建一个切片头(一个包含三个字段的结构体:指向底层数组的指针、当前长度

len

和容量

cap

),这个切片头通常会在栈上分配(如果切片变量没有逃逸到堆)。所以,

make

不仅分配了数据存储空间,还构建了访问和管理这块空间的“控制器”。映射 (Map):

make(map[K]V)

的底层实现更为复杂。它会分配一个哈希表结构,这包括一系列的桶(buckets,用于存储键值对)以及一些管理哈希表状态的元数据。这些结构会在堆上分配,并且

make

会执行一系列初始化操作,比如设置哈希函数、负载因子阈值等,以确保哈希表在插入第一个元素之前是有效且可用的。这远超简单的内存清零。通道 (Channel):

make(chan T, bufferSize)

同样涉及复杂的初始化。它会在堆上分配一个环形缓冲区(如果

bufferSize > 0

),以及用于同步的互斥锁(mutex)和条件变量(cond vars)。

make

会初始化这些同步原语和缓冲区,使其能够安全地进行并发读写操作。

可以说,

new

是内存分配的“原子操作”,而

make

则是针对特定Go内置复合类型的“构造函数”,它封装了更复杂的内存分配和内部状态设置。

Golang中make和new与变量声明的关联与最佳实践

理解

make

new

,也需要把它们和Go语言中常规的变量声明方式联系起来看。这三者共同构成了Go中内存和变量管理的主要手段。

当你使用

var x Type

声明一个变量时,Go会根据

Type

的零值来初始化

x

。如果

Type

是一个值类型(如

int

,

bool

,

struct

),

x

会直接在栈上(或经过逃逸分析后在堆上)分配并初始化为零值。例如,

var i int

会得到

i=0

var s MyStruct

会得到一个所有字段都为零值的

MyStruct

实例。然而,如果

Type

是切片、映射或通道,它们的零值是

nil

。这意味着

var mySlice []int

会得到一个

nil

切片,它没有底层数组,不能直接进行索引访问(虽然

append

操作可以),尝试访问其元素会引发运行时错误。同样,

var myMap map[string]int

var myChan chan int

也都是

nil

,不能直接使用。

这就是

make

发挥作用的地方。对于切片、映射和通道,如果你打算立即使用它们(比如向切片中添加元素,向映射中插入键值对,或者在通道上发送/接收数据),那么务必使用

make

进行初始化。这是最佳实践,它能保证这些数据结构是可用的,避免运行时错误。

// 错误示范:nil map不能直接赋值// var m map[string]int// m["key"] = 1 // panic: assignment to entry in nil map// 正确使用make初始化mapm := make(map[string]int)m["key"] = 1 // OK

new

则更常用于获取一个指向自定义结构体零值的指针。虽然你也可以

var p *MyStruct

,但这样

p

会是

nil

,你还需要

p = &MyStruct{}

或者

p = new(MyStruct)

来分配内存。

new(MyStruct)

直接给你一个非

nil

*MyStruct

,其所有字段都已初始化为零值。我个人在创建结构体实例时,如果需要一个指针,更倾向于使用复合字面量

&MyStruct{}

,因为它允许我在创建的同时初始化字段,代码可读性更好。但

new

在某些场景下,比如泛型编程中需要一个通用指针时,依然有其独特的价值。

总结来说,

make

是Go语言中为三种特定内建复合类型量身定制的“工厂”,确保它们在创建时就具备完整功能。而

new

则是一个通用的“内存分配器”,它只负责提供一块清零的内存空间,并返回指向它的指针。理解它们各自的职责和使用场景,是写出健壮Go代码的关键一步。

以上就是Golang的make和new函数有什么区别 对比内存分配方式的底层差异的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1398176.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
Go并发编程:使用Channel实现Goroutine间的通信
上一篇 2025年12月15日 15:16:20
如何在Golang中集成Nix包管理器 详解可复现开发环境配置方法
下一篇 2025年12月15日 15:16:35

相关推荐

  • composer require-dev和require有什么不同_Composer Require与Require-Dev区别解析

    require用于声明项目运行必需的依赖,如框架、数据库组件和第三方SDK,这些包会随项目部署到生产环境;2. require-dev用于声明仅在开发和测试阶段需要的工具,如PHPUnit、PHPStan、Faker等,不会默认部署到生产环境;3. 安装时composer install根据环境决定…

    2026年5月10日
    1000
  • Golang JSON序列化:控制敏感字段暴露的最佳实践

    本教程探讨golang中如何高效控制结构体字段在json序列化时的可见性。当需要将包含敏感信息的结构体数组转换为json响应时,通过利用`encoding/json`包提供的结构体标签,特别是`json:”-“`,可以轻松实现对特定字段的忽略,从而避免敏感数据泄露,确保api…

    2026年5月10日
    300
  • 比特币新手教程 比特币交易平台有哪些

    比特币是一种去中心化的数字货币,基于区块链技术实现点对点交易,具有匿名性、有限发行和不可篡改等特点;新手可通过交易所购买,P2P交易获得比特币,常用平台包括Binance、OKX和Huobi;交易流程包括注册账户、实名认证、绑定支付方式、充值法币并下单购买,可选择市价单或限价单;比特币存储方式有交易…

    2026年5月10日
    000
  • Golang gRPC流式请求异常处理

    在Golang的gRPC流式通信中,必须通过context.Context处理异常。应监听上下文取消或超时,及时释放资源,设置合理超时,避免连接长时间挂起,并在goroutine中通过context控制生命周期。 在使用 Golang 和 gRPC 实现流式通信时,异常处理是确保服务健壮性的关键部分…

    2026年5月10日
    000
  • Go语言mgo查询构建:深入理解bson.M与日期范围查询的正确实践

    本文旨在解决go语言mgo库中构建复杂查询时,特别是涉及嵌套`bson.m`和日期范围筛选的常见错误。我们将深入剖析`bson.m`的类型特性,解释为何直接索引`interface{}`会导致“invalid operation”错误,并提供一种推荐的、结构清晰的代码重构方案,以确保查询条件能够正确…

    2026年5月10日
    100
  • 修复点击时按钮抖动:CSS垂直对齐实践

    本文探讨了在Web开发中,交互式按钮(如播放/暂停按钮)在点击时发生意外垂直位移的问题。通过分析CSS样式变化对元素布局的影响,我们发现这是由于按钮不同状态下的边框样式和内边距改变,以及默认的垂直对齐行为共同作用所致。核心解决方案是利用CSS的vertical-align属性,将其设置为middle…

    2026年5月10日
    100
  • 理解编程指令:当结果正确,但实现方式不符要求时

    本文探讨了在编程实践中,即使程序输出了正确的结果,但若其实现方式未能严格遵循既定指令,仍可能被视为“不正确”的问题。我们将通过具体示例,对比直接求和与累加求和两种实现策略,强调理解和遵守编程规范的重要性,以确保代码的健壮性、可维护性及符合项目要求。 在软件开发过程中,我们经常会遇到这样的情况:编写的…

    2026年5月10日
    000
  • Golang goroutine与channel调试技巧

    使用go run -race检测数据竞争,结合runtime.NumGoroutine监控协程数量,通过pprof分析阻塞调用栈,利用select超时避免永久阻塞,有效排查goroutine泄漏、死锁和数据竞争问题。 Go语言的goroutine和channel是并发编程的核心,但它们也带来了调试上…

    2026年5月10日
    000
  • php常量怎么用_PHP常量(define/const)定义与使用方法

    PHP中可通过define函数和const关键字定义常量,用于存储不可变值。define适用于全局作用域,支持动态名称和条件定义,如define(‘SITE_NAME’, ‘MyWebsite’);const在编译时生效,语法简洁但限制多,只能在类或全…

    2026年5月10日
    000
  • 创建指定大小并填充特定数据的Golang文件教程

    本文将介绍如何使用Golang创建一个指定大小的文件,并用特定数据填充它。我们将使用 `os` 包提供的函数来创建和截断文件,从而实现快速生成大文件的目的。示例代码展示了如何创建一个10MB的文件,并将其填充为全零数据。掌握这些方法,可以方便地在例如日志系统或磁盘队列等场景中,预先创建测试文件或初始…

    2026年5月10日
    000
  • Golang空接口如何应用在项目中

    空接口可用于接收任意类型值,常见于日志函数、通用数据结构、JSON动态解析及配置驱动逻辑,提升代码灵活性,但需配合类型断言确保安全,避免滥用以降低维护成本。 空接口 interface{} 在 Go 语言中是一个非常灵活的类型,它可以存储任何类型的值。虽然它牺牲了一部分类型安全,但在实际项目中合理使…

    2026年5月10日
    300
  • Go语言网络编程入门:构建TCP客户端/服务器

    本文旨在为Go语言初学者提供一份简洁明了的网络编程入门指南,重点介绍如何使用TCP套接字构建简单的客户端/服务器应用。通过示例代码和注意事项,帮助读者快速上手Go语言的网络编程,并了解一些最佳实践。 Go语言对网络编程提供了强大的支持,通过标准库net包,可以轻松实现各种网络应用。本文将重点介绍如何…

    2026年5月10日
    000
  • Golang使用Protobuf定义接口与消息格式

    Protobuf通过字段编号实现兼容性,新增字段可忽略、删除字段可保留编号,确保新旧版本互操作,支持服务独立演进。 在Golang项目中,利用Protobuf定义接口和消息格式,本质上是为服务间通信构建了一套高效、类型安全且跨语言的契约。它让数据结构清晰可见,RPC调用标准化,极大地简化了分布式系统…

    2026年5月10日
    000
  • Go语言接口与切片:如何识别和操作[]interface{}

    本文将深入探讨Go语言中如何识别和操作`[]interface{}`类型的切片。我们将介绍类型断言(Type Assertion)的关键作用,并通过`switch`语句演示如何安全地检测`[]interface{}`类型,并进而遍历其内部元素。文章旨在提供清晰的示例代码和专业指导,帮助开发者有效地处…

    2026年5月10日
    300
  • 如何在Golang中声明指针变量 使用&和*操作符示例

    答案是:Go中指针通过&取地址和解引用操作实现对变量地址的访问与值的修改,声明格式为Type,初始值为nil,常用于函数传参和内存优化。 在Golang中,指针变量用于存储另一个变量的内存地址。通过使用 & 和 * 操作符,可以获取变量地址和访问指针指向的值。下面详细介绍如何声明指针…

    2026年5月10日
    000
  • JavaScript 高效判断页面所有复选框状态的技巧与实践

    本文旨在提供一套高效且专业的javascript方法,用于判断网页中所有复选框的选中状态。我们将探讨如何利用`array.some()`快速确定是否有未选中的复选框(进而判断是否全部选中),以及如何使用`array.filter()`统计选中和未选中的复选框数量。通过优化dom元素选择和数组操作,提…

    2026年5月10日
    100
  • GolangWeb项目异常捕获与日志记录

    答案:通过中间件使用defer和recover捕获panic,结合zap等结构化日志库记录请求链路信息,为每个请求生成trace ID,实现异常捕获与可追踪日志,提升系统稳定性与可观测性。 在Go语言Web项目中,异常捕获与日志记录是保障系统稳定性和可维护性的关键环节。Go本身没有像其他语言那样的t…

    2026年5月10日
    100
  • 硬盘数据被误删除怎么办?教你快速找回删除的文件!

    硬盘数据被误删除,别慌!恢复数据并非不可能,关键在于你接下来的操作。立刻停止对该硬盘的任何写入操作,然后尝试使用专业的数据恢复软件。 解决方案 首先,数据恢复的原理是,删除文件后,操作系统只是将文件占用的空间标记为“可覆盖”,但文件本身的数据可能还存在于硬盘上。所以,避免新的数据写入覆盖掉旧数据,是…

    2026年5月10日
    000
  • Golang如何优化日志写入性能_Golang日志写入与文件IO优化方法

    使用缓冲、异步写入、高性能日志库和优化IO策略提升Golang日志性能,推荐zap+异步缓冲+SSD组合以平衡实时性、可靠性与高并发需求。 在高并发场景下,Golang程序的日志写入可能成为性能瓶颈。频繁的文件IO操作不仅影响响应速度,还可能导致系统负载升高。要提升日志写入性能,不能只依赖简单的fm…

    2026年5月10日
    300
  • 控制HTML Canvas颜色空间输出24位深度TIFF图像

    本教程详细介绍了如何在web前端环境中,特别是结合`html2canvas`和`canvas-to-tiff`库时,通过明确设置html canvas的颜色空间为`srgb`,从而确保输出24位深度的tiff图像。文章将提供具体的javascript代码示例,并解释其原理,帮助开发者解决canvas…

    2026年5月10日
    200

发表回复

登录后才能评论
关注微信