Go中的new()和make()函数区别及底层原理详解

 更新时间:2023年09月24日 15:26:20   作者:TimLiu  
这篇文章主要为大家介绍了Go中的new()和make()函数区别及底层原理详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪

引言

在 Go 语言中,new() 和 make() 是两个常用的函数,用于创建和初始化不同类型的变量。本文将深入探讨 new() 和 make() 的区别、使用场景以及底层实现原理。

  • Go 中的 new() 和 make() 函数是用于创建和初始化变量的重要工具。
  • new() 用于创建指定类型的零值变量,并返回该变量的指针。
  • make() 用于创建并初始化引用类型的变量,如切片、映射和通道。

new() 函数

  • new() 函数的基本语法及用法。
  • new() 创建的变量是指定类型的零值,并返回该变量的指针。
  • new() 适用于创建引用类型以外的其他类型变量。
package main
import "fmt"
func main() {
    // 使用 new() 创建一个 int 类型的零值变量的指针
    numPtr := new(int)
    fmt.Println(*numPtr) // 输出 0
}

make() 函数

  • make() 函数的基本语法及用法。
  • make() 用于创建并初始化引用类型的变量。
  • make() 适用于创建切片、映射和通道等引用类型的变量。
  • make() 创建的变量不是零值,而是根据类型进行初始化。
package main
import "fmt"
func main() {
    // 使用 make() 创建一个切片,并初始化长度为 3 的切片
    slice := make([]int, 3)
    fmt.Println(slice) // 输出 [0 0 0]
}

new() 和 make() 的区别

  • new() 用于创建任意类型的变量,而 make() 仅用于创建引用类型的变量。
  • new() 返回的是指针,而 make() 返回的是初始化后的值。
  • new() 创建的变量是零值,make() 创建的变量是根据类型进行初始化。
package main
import "fmt"
func main() {
    // 使用 new() 创建一个结构体的指针
    personPtr := new(Person)
    personPtr.Name = "Alice"
    personPtr.Age = 30
    fmt.Println(personPtr) // 输出 &{Alice 30}
    // 使用 make() 创建一个映射,并初始化键值对
    m := make(map[string]int)
    m["one"] = 1
    m["two"] = 2
    fmt.Println(m) // 输出 map[one:1 two:2]
}
type Person struct {
    Name string
    Age  int
}

new() 和 make() 的底层实现原理

在 Go 语言中,new() 和 make() 的底层实现原理略有不同。

new() 的底层实现原理

  • new() 函数在底层使用了 Go 的 runtime.newobject 函数。
  • runtime.newobject 函数会分配一块内存,大小为指定类型的大小,并将该内存清零。
  • 然后,runtime.newobject 函数会返回这块内存的指针。

下面是 new() 函数的简化版本的底层实现原理示例代码:

package main
import (
    "fmt"
    "unsafe"
)
func main() {
    // 使用 new() 创建一个 int 类型的零值变量的指针
    numPtr := new(int)
    // 获得指针的值
    ptrValue := uintptr(unsafe.Pointer(numPtr))
    // 输出指针的值
    fmt.Println(ptrValue)
}

在上述示例代码中,我们使用了 unsafe 包中的 Pointer 和 uintptr 类型来操作指针。我们首先使用 new(int) 创建一个 int 类型的零值变量的指针 numPtr,然后通过 unsafe.Pointer 将指针转换为 unsafe.Pointer 类型,再通过 uintptr 将 unsafe.Pointer 值转换为 uintptr 类型,最后输出指针的值。这个值就是我们所创建的变量的内存地址。

make() 的底层实现原理

  • make() 函数在底层使用了 Go 的 runtime.makesliceruntime.makemap 和 runtime.makechan 函数。
  • runtime.makeslice 函数用于创建切片,它会分配一块连续的内存空间,并返回切片结构体。
  • runtime.makemap 函数用于创建映射,它会分配一块哈希表内存,并返回映射结构体。
  • runtime.makechan 函数用于创建通道,它会分配一块通道内存,并返回通道结构体。

下面是 make() 函数的简化版本的底层实现原理示例代码:

package main
import (
    "fmt"
    "reflect"
    "unsafe"
)
func main() {
    // 使用 make() 创建一个切片,并初始化长度为 3 的切片
    slice := make([]int, 3)
    // 获得切片的值和长度
    sliceValue := reflect.ValueOf(slice)
    sliceData := sliceValue.Elem().UnsafeAddr()
    sliceLen := sliceValue.Len()
    // 输出切片的值和长度
    fmt.Println(sliceData, sliceLen)
}

在上述示例代码中,我们使用了 reflect 包中的 ValueElem 和 UnsafeAddr 方法来操作切片。我们首先使用 make([]int, 3) 创建一个长度为 3 的切片 slice,然后通过 reflect.ValueOf 将切片转换为 reflect.Value 类型,再通过 Elem 方法获取切片的元素,并通过 UnsafeAddr 方法获取切片的底层数组的指针,最后通过 Len 方法获取切片的长度。这样,我们就可以获得切片的底层数组的指针和长度。

请注意,上述示例代码中使用了 reflect 和 unsafe 包,这是为了演示 make() 的底层实现原理而引入的,实际开发中并不需要经常使用这些包。

总结

通过深入了解 new() 和 make() 函数的区别、使用场景以及底层实现原理,读者可以更好地理解和运用这两个函数,并完美解决掉面试官的问题,并在实际开发中做出准确的选择。

以上就是Go中的new()和make()函数区别及底层原理详解的详细内容,更多关于Go new make函数的资料请关注脚本之家其它相关文章!

相关文章

  • Go中Context使用源码解析

    Go中Context使用源码解析

    这篇文章主要为大家介绍了Go中Context使用源码解析,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-04-04
  • 详解Golang time包中的结构体time.Ticker

    详解Golang time包中的结构体time.Ticker

    在日常开发过程中,会频繁遇到对时间进行操作的场景,使用 Golang 中的 time 包可以很方便地实现对时间的相关操作,接下来的几篇文章会详细讲解 time 包,本文讲解一下 time 包中的结构体 time.Ticker,需要的朋友可以参考下
    2023-08-08
  • 一文搞懂如何实现Go 超时控制

    一文搞懂如何实现Go 超时控制

    这篇文章主要介绍了一文搞懂如何实现Go 超时控制,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2021-03-03
  • Golang实现验证一个字符串是否为URL

    Golang实现验证一个字符串是否为URL

    在实际开发过程中,有时候会遇到 URL 的校验问题,Go 语言中有哪些方法去验证一个字符串是否满足 URL 格式呢?本文就来和大家详细讲讲
    2023-04-04
  • golang强制类型转换和类型断言

    golang强制类型转换和类型断言

    这篇文章主要介绍了详情介绍golang类型转换问题,分别由介绍类型断言和类型转换,这两者都是不同的概念,下面文章围绕类型断言和类型转换的相关资料展开文章的详细内容,需要的朋友可以参考以下
    2021-12-12
  • Golang线程池与协程池的使用

    Golang线程池与协程池的使用

    在Golang中,线程池和协程池是非常常见且重要的概念,它们可以提高应用程序的并发处理能力和性能,减少资源的浪费,本文就来介绍一下Golang线程池与协程池的使用,感兴趣的可以了解一下
    2024-04-04
  • Go设计模式之单例模式图文详解

    Go设计模式之单例模式图文详解

    单例模式是一种创建型设计模式,让你能够保证一个类只有一个实例,并提供一个访问该实例的全局节点,本文就通过图文给大家介绍一下Go的单例模式,需要的朋友可以参考下
    2023-07-07
  • Go中如何使用set的方法示例

    Go中如何使用set的方法示例

    这篇文章主要介绍了Go中如何使用set的方法示例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2019-09-09
  • 基于Go语言实现高性能文件上传下载系统

    基于Go语言实现高性能文件上传下载系统

    在Web应用开发中,文件上传下载是一个非常常见的需求,本文将介绍如何使用Go语言实现一个安全、高效的本地文件存储系统,感兴趣的小伙伴可以了解下
    2025-03-03
  • Go并发之RWMutex的源码解析详解

    Go并发之RWMutex的源码解析详解

    RWMutex是一个支持并行读串行写的读写锁。RWMutex具有写操作优先的特点,写操作发生时,仅允许正在执行的读操作执行,后续的读操作都会被阻塞。本文就来从源码解析一下RWMutex的使用
    2023-03-03

最新评论