0

0

Go语言实现System V共享内存的完整教程

碧海醫心

碧海醫心

发布时间:2026-01-24 15:26:33

|

770人浏览过

|

来源于php中文网

原创

Go语言实现System V共享内存的完整教程

本文介绍如何在go中通过系统调用原生支持system v共享内存(shm),避免c++go指针管理风险,实现与c/c++等传统ipc程序高效协同——适用于大块数据零拷贝交互场景。

在Go生态中,“通过通信共享内存”(Do not communicate by sharing memory; instead, share memory by communicating)是核心设计哲学,但这并不意味着Go无法支持传统共享内存IPC。当需要与遗留系统(如问题中所述的应用A)协同工作,且数据量巨大(数十MB甚至GB级)、对性能与延迟敏感时,System V共享内存(shmget/shmat/shmdt/shmctl)仍是Linux下最直接、零拷贝的解决方案。

幸运的是,Go标准库虽不直接封装shm*系列系统调用,但通过官方维护的 golang.org/x/sys/unix 包,可安全、高效地调用底层UNIX系统调用,完全无需CGO——这正是规避C指针跨语言传递风险(如原示例中C.free()崩溃的根本原因)的最佳实践。

✅ 正确做法:纯Go调用System V共享内存

以下是一个完整的、生产就绪的Go共享内存客户端示例(对应问题中的程序B),用于连接已有key(如0x1234)创建的共享内存段,并安全读写:

package main

import (
    "fmt"
    "unsafe"

    "golang.org/x/sys/unix"
)

const (
    shmKey   = 0x1234          // 与应用A约定的ftok key或IPC_PRIVATE派生key
    shmSize  = 1024 * 1024 * 64 // 64MB,按需调整
    shmFlag  = unix.IPC_CREAT | 0666
)

func main() {
    // 1. 获取共享内存标识符(若不存在则创建)
    shmid, err := unix.Shmget(shmKey, shmSize, shmFlag)
    if err != nil {
        panic(fmt.Sprintf("shmget failed: %v", err))
    }
    fmt.Printf("Shared memory ID: %d\n", shmid)

    // 2. 映射到当前进程地址空间(只读或读写)
    addr, err := unix.Shmat(shmid, nil, 0)
    if err != nil {
        panic(fmt.Sprintf("shmat failed: %v", err))
    }
    defer func() {
        if err := unix.Shmdt(addr); err != nil {
            fmt.Printf("WARNING: shmdt failed: %v\n", err)
        }
    }()
    fmt.Printf("Shared memory attached at address: %p\n", addr)

    // 3. 安全访问内存:转换为Go切片(零拷贝!)
    // 注意:必须确保应用A已写入有效数据,且有同步机制(如信号量/文件锁)
    data := (*[1 << 30]byte)(unsafe.Pointer(addr))[:shmSize:shmSize]

    // 示例:读取前100字节(假设应用A已写入)
    fmt.Printf("First 100 bytes (hex): %x\n", data[:100])

    // 示例:写回处理结果(如状态码)到前4字节
    *(*uint32)(unsafe.Pointer(&data[0])) = 0xDEADBEEF

    // 4. (可选)显式控制段生命周期(如通知A处理完成)
    // unix.Shmctl(shmid, unix.IPC_STAT, &buf) 等
}
? 关键要点说明:unix.Shmget 返回内核共享内存段ID,非地址;unix.Shmat 才返回映射后的虚拟地址(uintptr)。使用 (*[1

⚠️ 必须配套的同步机制

共享内存本身不提供同步。应用A与程序B必须约定额外的同步原语:

SpeechEasy
SpeechEasy

SpeechEasy是一种合成语音解决方案,可以让用户从文本生成高质量、易于理解的音频。

下载

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

  • 推荐:使用 unix.Semget / unix.Semop 实现POSIX信号量(同属x/sys/unix包);
  • 替代:基于文件的flock、inotify事件,或更现代的eventfd(需unix.Eventfd);
  • 严禁:仅靠sleep轮询——既低效又不可靠。

? 总结与最佳实践

事项 建议
是否用CGO? ❌ 否。x/sys/unix 提供完备、安全、无CGO依赖的系统调用封装。
指针传递 ❌ 绝不传递C指针给Go或反之;所有内存操作通过unsafe.Slice或(*T)(unsafe.Pointer())完成。
错误处理 ✅ 每个unix.*调用后检查err,System V IPC失败返回明确errno(如unix.EEXIST, unix.ENOENT)。
跨平台性 ⚠️ System V shm为Linux/Unix特有;如需跨平台,请评估mmap+文件-backed方案(unix.Mmap)。
替代方案 ? 对新项目,优先考虑Go原生通道(chan)+ net/rpc 或 gRPC;仅对存量IPC集成选用共享内存。

通过以上方式,你不仅能安全、高效地让Go程序B接入应用A的共享内存管道,还能彻底规避CGO带来的内存模型混乱与运行时崩溃风险——真正践行“用Go的方式,解决系统级问题”。

相关专题

更多
golang如何定义变量
golang如何定义变量

golang定义变量的方法:1、声明变量并赋予初始值“var age int =值”;2、声明变量但不赋初始值“var age int”;3、使用短变量声明“age :=值”等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

180

2024.02.23

golang有哪些数据转换方法
golang有哪些数据转换方法

golang数据转换方法:1、类型转换操作符;2、类型断言;3、字符串和数字之间的转换;4、JSON序列化和反序列化;5、使用标准库进行数据转换;6、使用第三方库进行数据转换;7、自定义数据转换函数。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

228

2024.02.23

golang常用库有哪些
golang常用库有哪些

golang常用库有:1、标准库;2、字符串处理库;3、网络库;4、加密库;5、压缩库;6、xml和json解析库;7、日期和时间库;8、数据库操作库;9、文件操作库;10、图像处理库。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

341

2024.02.23

golang和python的区别是什么
golang和python的区别是什么

golang和python的区别是:1、golang是一种编译型语言,而python是一种解释型语言;2、golang天生支持并发编程,而python对并发与并行的支持相对较弱等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

209

2024.03.05

golang是免费的吗
golang是免费的吗

golang是免费的。golang是google开发的一种静态强类型、编译型、并发型,并具有垃圾回收功能的开源编程语言,采用bsd开源协议。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

393

2024.05.21

golang结构体相关大全
golang结构体相关大全

本专题整合了golang结构体相关大全,想了解更多内容,请阅读专题下面的文章。

200

2025.06.09

golang相关判断方法
golang相关判断方法

本专题整合了golang相关判断方法,想了解更详细的相关内容,请阅读下面的文章。

191

2025.06.10

golang数组使用方法
golang数组使用方法

本专题整合了golang数组用法,想了解更多的相关内容,请阅读专题下面的文章。

293

2025.06.17

c++ 根号
c++ 根号

本专题整合了c++根号相关教程,阅读专题下面的文章了解更多详细内容。

24

2026.01.23

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
PostgreSQL 教程
PostgreSQL 教程

共48课时 | 7.7万人学习

Git 教程
Git 教程

共21课时 | 2.9万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号