搜索
首页后端开发Golang解密Go:空结构

解密Go:空结构

Sep 11, 2024 pm 12:30 PM

Decrypt Go: empty struct

在 Go 中,普通的结构体通常会占用一块内存。但是,有一种特殊情况:如果它是空结构,则其大小为零。这怎么可能,空结构有什么用?

本文首发于Medium MPP计划。如果您是Medium用户,请在Medium上关注我。非常感谢。

type Test struct {
    A int
    B string
}

func main() {
    fmt.Println(unsafe.Sizeof(new(Test)))
    fmt.Println(unsafe.Sizeof(struct{}{}))
}

/*
8
0
*/

空结构的秘密

特殊变量:zerobase

空结构是没有内存大小的结构。这种说法是正确的,但更准确地说,它实际上有一个特殊的起点:zerobase 变量。这是一个uintptr全局变量,占用8个字节。每当定义无数个 struct {} 变量时,编译器都会分配这个零基变量的地址。换句话说,在 Go 中,任何大小为 0 的内存分配都使用相同的地址 &zerobase。

示例

package main

import "fmt"

type emptyStruct struct {}

func main() {
    a := struct{}{}
    b := struct{}{}
    c := emptyStruct{}

    fmt.Printf("%p\n", &a)
    fmt.Printf("%p\n", &b)
    fmt.Printf("%p\n", &c)
}

// 0x58e360
// 0x58e360
// 0x58e360

空结构体的变量的内存地址都是相同的。这是因为编译器在编译过程中遇到这种特殊类型的内存分配时会分配&zerobase。这个逻辑在mallocgc函数中:

//go:linkname mallocgc  
func mallocgc(size uintptr, typ *_type, needzero bool) unsafe.Pointer {  
    ...
    if size == 0 {  
       return unsafe.Pointer(&zerobase)  
    }
    ...

这是 Empty 结构的秘密。通过这个特殊的变量,我们可以完成很多功能。

空结构和内存对齐

通常,如果空结构是较大结构的一部分,它不会占用内存。但是,有一种特殊情况,即空结构是最后一个字段;它会触发内存对齐。

示例

type A struct {
    x int
    y string
    z struct{}
}
type B struct {
    x int
    z struct{}
    y string
}

func main() {
    println(unsafe.Alignof(A{}))
    println(unsafe.Alignof(B{}))
    println(unsafe.Sizeof(A{}))
    println(unsafe.Sizeof(B{}))
}

/**
8
8
32
24
**/

当存在指向字段的指针时,返回的地址可能位于结构体之外,如果在释放结构体时未释放内存,则可能导致内存泄漏。因此,当空结构是另一个结构的最后一个字段时,为了安全起见,会分配额外的内存。如果空结构体位于开头或中间,则其地址与下一个变量相同。

type A struct {  
    x int  
    y string  
    z struct{}  
}  
type B struct {  
    x int  
    z struct{}  
    y string  
}  

func main() {  
    a := A{}  
    b := B{}  
    fmt.Printf("%p\n", &a.y)  
    fmt.Printf("%p\n", &a.z)  
    fmt.Printf("%p\n", &b.y)  
    fmt.Printf("%p\n", &b.z)  
}

/**
0x1400012c008
0x1400012c018
0x1400012e008
0x1400012e008
**/

空结构的用例

空struct struct{}存在的核心原因是为了节省内存。当您需要一个结构体但不关心其内容时,请考虑使用空结构体。 Go的核心复合结构如map、chan、slice都可以使用struct{}。

映射和结构{}

// Create map
m := make(map[int]struct{})
// Assign value
m[1] = struct{}{}
// Check if key exists
_, ok := m[1]

陈和结构{}

一个经典场景结合了channel和struct{},其中struct{}通常用作信号而不关心其内容。正如前面文章分析的那样,通道的本质数据结构是管理结构加上环形缓冲区。如果 struct{} 用作元素,则环形缓冲区为零分配。

chan 和 struct{} 一起使用的唯一用途是信号传输,因为空结构本身不能携带任何值。一般来说,它是在没有缓冲通道的情况下使用的。

// Create a signal channel
waitc := make(chan struct{})

// ...
goroutine 1:
    // Send signal: push element
    waitc 



<p>在这种情况下,struct{} 是绝对必要的吗?事实并非如此,节省的内存可以忽略不计。关键是不关心chan的元素值,所以使用struct{}。</p>

<h3>
  
  
  概括
</h3>

<ol>
<li>空结构体仍然是一个结构体,只是大小为 0。</li>
<li>所有空结构共享相同的地址:zerobase 的地址。</li>
<li>我们可以利用空结构体不占用内存的特性来优化代码,比如使用map来实现集合和通道。</li>
</ol>

<h3>
  
  
  参考
</h3>

<ol>
<li>空结构,戴夫·切尼</li>
<li>Go 最细节篇— struct{} 空结构体到底啥是啥?</li>
</ol>


          

            
        

以上是解密Go:空结构的详细内容。更多信息请关注PHP中文网其他相关文章!

声明
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn
实施静音和锁以寻求线程安全性实施静音和锁以寻求线程安全性May 05, 2025 am 12:18 AM

在Go中,使用互斥锁和锁是确保线程安全的关键。1)使用sync.Mutex进行互斥访问,2)使用sync.RWMutex处理读写操作,3)使用原子操作进行性能优化。掌握这些工具及其使用技巧对于编写高效、可靠的并发程序至关重要。

基准测试和分析并发GO代码基准测试和分析并发GO代码May 05, 2025 am 12:18 AM

如何优化并发Go代码的性能?使用Go的内置工具如gotest、gobench和pprof进行基准测试和性能分析。1)使用testing包编写基准测试,评估并发函数的执行速度。2)通过pprof工具进行性能分析,识别程序中的瓶颈。3)调整垃圾收集设置以减少其对性能的影响。4)优化通道操作和限制goroutine数量以提高效率。通过持续的基准测试和性能分析,可以有效提升并发Go代码的性能。

并发程序中的错误处理:避免常见的陷阱并发程序中的错误处理:避免常见的陷阱May 05, 2025 am 12:17 AM

避免并发Go程序中错误处理的常见陷阱的方法包括:1.确保错误传播,2.处理超时,3.聚合错误,4.使用上下文管理,5.错误包装,6.日志记录,7.测试。这些策略有助于有效处理并发环境中的错误。

隐式接口实现:鸭打字的力量隐式接口实现:鸭打字的力量May 05, 2025 am 12:14 AM

IndimitInterfaceImplementationingingoembodiesducktybybyallowingTypestoSatoSatiSatiSatiSatiSatiSatsatSatiSatplicesWithouTexpliclIctDeclaration.1)itpromotesflemotesflexibility andmodularitybybyfocusingion.2)挑战挑战InclocteSincludeUpdatingMethodSignateSignatiSantTrackingImplections.3)工具li

进行错误处理:最佳实践和模式进行错误处理:最佳实践和模式May 04, 2025 am 12:19 AM

在Go编程中,有效管理错误的方法包括:1)使用错误值而非异常,2)采用错误包装技术,3)定义自定义错误类型,4)复用错误值以提高性能,5)谨慎使用panic和recover,6)确保错误消息清晰且一致,7)记录错误处理策略,8)将错误视为一等公民,9)使用错误通道处理异步错误。这些做法和模式有助于编写更健壮、可维护和高效的代码。

您如何在GO中实施并发?您如何在GO中实施并发?May 04, 2025 am 12:13 AM

在Go中实现并发可以通过使用goroutines和channels来实现。1)使用goroutines来并行执行任务,如示例中同时享受音乐和观察朋友。2)通过channels在goroutines之间安全传递数据,如生产者和消费者模式。3)避免过度使用goroutines和死锁,合理设计系统以优化并发程序。

在GO中构建并发数据结构在GO中构建并发数据结构May 04, 2025 am 12:09 AM

Gooffersmultipleapproachesforbuildingconcurrentdatastructures,includingmutexes,channels,andatomicoperations.1)Mutexesprovidesimplethreadsafetybutcancauseperformancebottlenecks.2)Channelsofferscalabilitybutmayblockiffullorempty.3)Atomicoperationsareef

将GO的错误处理与其他编程语言进行比较将GO的错误处理与其他编程语言进行比较May 04, 2025 am 12:09 AM

go'serrorhandlingisexplicit,治疗eRROSASRETRATERTHANEXCEPTIONS,与pythonandjava.1)go'sapphifeensuresererrawaresserrorawarenessbutcanleadtoverbosecode.2)pythonandjavauseexeexceptionseforforforforforcleanerCodebutmaymobisserrors.3)

See all articles

热AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智能驱动的应用程序,用于创建逼真的裸体照片

AI Clothes Remover

AI Clothes Remover

用于从照片中去除衣服的在线人工智能工具。

Undress AI Tool

Undress AI Tool

免费脱衣服图片

Clothoff.io

Clothoff.io

AI脱衣机

Video Face Swap

Video Face Swap

使用我们完全免费的人工智能换脸工具轻松在任何视频中换脸!

热工具

记事本++7.3.1

记事本++7.3.1

好用且免费的代码编辑器

DVWA

DVWA

Damn Vulnerable Web App (DVWA) 是一个PHP/MySQL的Web应用程序,非常容易受到攻击。它的主要目标是成为安全专业人员在合法环境中测试自己的技能和工具的辅助工具,帮助Web开发人员更好地理解保护Web应用程序的过程,并帮助教师/学生在课堂环境中教授/学习Web应用程序安全。DVWA的目标是通过简单直接的界面练习一些最常见的Web漏洞,难度各不相同。请注意,该软件中

WebStorm Mac版

WebStorm Mac版

好用的JavaScript开发工具

SublimeText3 英文版

SublimeText3 英文版

推荐:为Win版本,支持代码提示!

SublimeText3 Mac版

SublimeText3 Mac版

神级代码编辑软件(SublimeText3)