0

0

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

花韻仙語

花韻仙語

发布时间:2026-01-24 11:38:20

|

829人浏览过

|

来源于php中文网

原创

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

本文介绍如何在go中通过系统调用原生支持system v共享内存(shm),无需c++go,兼容linux(如ubuntu 12.04),安全高效地与c/c++等遗留程序共享大块数据。

Go官方哲学强调“通过通信共享内存”(Share memory by communicating),但这并不排斥对底层共享内存机制的支持——尤其当需与已有System V IPC程序(如问题中的应用A)协同工作时,直接使用shmget/shmat/shmdt/shmctl等系统调用是合理且高效的方案。

幸运的是,Go标准生态已提供成熟、安全的系统调用封装:golang.org/x/sys/unix 包。它为Linux/Unix平台提供了完备、类型安全的系统调用接口,完全避免了CGO带来的指针跨语言传递风险(如原问题中C.free()崩溃的根本原因:返回/全局变量地址被误当作堆内存释放)。

以下是在Ubuntu 12.04+环境下,使用纯Go实现共享内存读写的完整示例:

墨鱼aigc
墨鱼aigc

一款超好用的Ai写作工具,为用户提供一键生成营销广告、原创文案、写作辅助等文字生成服务。

下载

✅ 正确做法:纯Go调用System V shm(无CGO)

package main

import (
    "fmt"
    "unsafe"

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

const (
    shmKey   = 0x12345678 // 与应用A约定的IPC key
    shmSize  = 1024 * 1024 // 1MB,按实际需求调整
    shmFlags = unix.IPC_CREAT | 0666
)

func main() {
    // 1. 创建或获取共享内存段
    shmid, err := unix.Shmget(shmKey, shmSize, shmFlags)
    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)
        }
    }()

    // 3. 安全访问内存:使用 unsafe.Slice(Go 1.17+)或手动转换
    data := (*[shmSize]byte)(addr)[:shmSize:shmSize]

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

    // 示例:写回结果(如状态码、处理长度等)
    copy(data[:8], []byte("SUCCESS!"))

    fmt.Println("Data processed and result written to shared memory.")
}

⚠️ 关键注意事项

  • 绝不混用CGO与非堆内存:原问题崩溃源于C函数返回全局数组buf地址,而Go错误调用C.free()试图释放该地址。C.free()仅适用于C.malloc/C.CString分配的内存。纯Go方案彻底规避此风险。
  • 显式分离生命周期:Shmat返回的地址必须配对调用Shmdt;若需长期驻留,应由主进程管理映射,子goroutine仅读写。
  • 同步至关重要:共享内存本身不提供同步机制。务必配合信号量(unix.Semget/unix.Semop)或文件锁(unix.Flock)协调A与B的读写顺序,防止竞态。
  • 权限与清理:首次创建时使用IPC_CREAT;生产环境建议在退出前调用unix.Shmctl(shmid, unix.IPC_RMID, nil)删除段,避免残留。
  • 跨平台提示:x/sys/unix主要面向Unix-like系统。Windows需改用syscall.CreateFileMapping等WinAPI(不在本文范围)。

✅ 总结

对于与遗留C程序通过System V共享内存协作的场景,优先采用golang.org/x/sys/unix包进行纯Go系统调用,而非CGO封装。它更安全、更可控、更易维护,且完全兼容Ubuntu 12.04及后续版本。只需三步:Shmget → Shmat → 安全切片访问,即可高效完成大数据量IPC。同步与清理逻辑需根据业务严格设计,这是共享内存正确性的基石。

立即学习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号