当前位置: 首页 > java >正文

《Go小技巧易错点100例》第三十一篇

本期分享:

1.Go struct内存对齐

2.使用空结构体(struct{})节省内存

Go struct内存对齐

在计算机系统中,CPU 访问内存时并不是逐字节读取的,而是以特定大小的块(通常为 4/8 字节)为单位进行读取。当数据的内存地址正好是其大小的整数倍时,称为自然对齐。Go 编译器会根据平台特性自动进行内存对齐优化,这种机制虽然可能产生填充字节,但能大幅提升内存访问效率。

内存对齐实战案例
// 未对齐的结构体 (24 bytes)
type BadStruct struct {a bool    // 1 byteb int64   // 8 bytesc bool    // 1 byte
}// 对齐优化后的结构体 (16 bytes)
type GoodStruct struct {b int64   // 8 bytes(偏移量 0)a bool    // 1 byte(偏移量 8)c bool    // 1 byte(偏移量 9)// 自动填充 6 bytes 到 16 bytes(满足 8 字节对齐)
}func main() {fmt.Println(unsafe.Sizeof(BadStruct{}))  // 输出 24fmt.Println(unsafe.Sizeof(GoodStruct{})) // 输出 16
}

优化原理

BadStructac 导致 b 需要填充 7 字节才能对齐

GoodStruct 通过字段排序减少填充,内存占用降低 33%

对齐规则总结

1)结构体整体大小需是最大字段对齐值的整数倍

2)每个字段的偏移量必须能整除其类型大小

3)嵌套结构体继承父结构体的对齐规则

使用空结构体(struct{})节省内存

struct{} 是 Go 语言中唯一零内存的类型:

fmt.Println(unsafe.Sizeof(struct{}{})) // 输出 0
六大应用场景

1)场景1:实现高效集合(Set)

type Set map[string]struct{}func (s Set) Add(key string) {s[key] = struct{}{}
}func (s Set) Contains(key string) bool {_, ok := s[key]return ok
}// 使用示例
s := make(Set)
s.Add("apple")
fmt.Println(s.Contains("apple")) // true

2)场景2:通道信号传递

func worker(stopCh <-chan struct{}) {for {select {case <-stopCh:returndefault:// 执行任务}}
}// 发送关闭信号
closeCh := make(chanstruct{})
go worker(closeCh)
close(closeCh) // 广播关闭

3)场景3:方法接收器(无状态)

type Logger struct{}func (Logger) Info(msg string) {fmt.Printf("[INFO] %s\n", msg)
}// 使用零内存接收器
var log Logger
log.Info("service started")

4)场景4:占位通道

// 限制并发数为 10
sem := make(chan struct{}, 10)
for i := 0; i < 1000; i++ {sem <- struct{}{}go func() {defer func() { <-sem }()// 业务逻辑}()
}

5)场景5:接口实现标记

type Marker interface {isMarker()
}type MyMarker struct{}func (MyMarker) isMarker() {}// 类型断言检查
func CheckMarker(v interface{}) bool {_, ok := v.(Marker)return ok
}

6)场景6:JSON 空对象

type Response struct {Data  interface{} `json:"data"`Error struct{}    `json:"error,omitempty"`
}// 序列化时自动忽略空 error 字段
resp := Response{Data: "success"}
jsonData, _ := json.Marshal(resp) // {"data":"success"}
注意事项

1)空结构体作为结构体字段时会产生对齐填充

type Wrapper struct {_ struct{} // 0 字节n int64    // 8 字节(偏移量 0)
}
fmt.Println(unsafe.Sizeof(Wrapper{})) // 8 bytes

2)不同地址的空结构体实例本质相同

a := struct{}{}
b := struct{}{}
fmt.Println(&a == &b) // 输出 true(编译器优化)

本篇结束~

http://www.xdnf.cn/news/5378.html

相关文章:

  • stm32week15
  • 轻量服务器与宝塔
  • 【递归、搜索与回溯算法】导论
  • 2025第九届御网杯网络安全大赛线上赛 区域赛WP (MISC和Crypto)(详解-思路-脚本)
  • [Java实战]Spring Boot 快速配置 HTTPS 并实现 HTTP 自动跳转(八)
  • Java反序列化漏洞
  • 第一章 初识Java
  • Kotlin Multiplatform--03:项目实战
  • 机器学习总结
  • C/C++实践(四)C++跨平台开发的系统性挑战与深度解决方案
  • 基于SpringBoot的小区停车位管理系统
  • 集合(1)
  • MATLAB中矩阵和数组的区别
  • Python-Venv多环境管理
  • JavaEE--文件操作和IO
  • cookie和session的区别
  • Qt开发经验 --- 避坑指南(14)
  • 【Linux篇】高并发编程终极指南:线程池优化、单例模式陷阱与死锁避坑实战
  • SpringBoot主入口类分析
  • 虚幻引擎5-Unreal Engine笔记之UE编辑器退出时的保存弹框
  • 【QT】UDP通讯本地调试
  • Pandas 时间处理利器:to_datetime() 与 Timestamp() 深度解析
  • 趣味编程:四叶草
  • Python赋能自动驾驶:如何打造高效的环境感知系统
  • 嵌入式硬件篇---TOF|PID
  • 微软向现实低头:悄悄延长Windows 10的Microsoft 365支持
  • 每日c/c++题 备战蓝桥杯(P1002 [NOIP 2002 普及组] 过河卒)
  • 数据仓库Hive
  • 【即插即用涨点模块】RFAConv感受野注意力卷积:突破卷积参数共享瓶颈,感受野注意力重塑空间特征提取【附源码】
  • 深度剖析多模态大模型中的视频编码器算法