在Go语言中调用C语言共享库的方法
珍惜时间,勤奋学习!今天给大家带来《在Go语言中调用C语言共享库的方法》,正文内容主要涉及到等等,如果你正在学习Golang,或者是对Golang有疑问,欢迎大家关注我!后面我会持续更新相关内容的,希望都能帮到正在学习的大家!
我使用 go 编写了 c 共享库。
package main import "c" import "log" //export runlib func runlib() { log.println("call runlib") } func init() { log.println("call init") } func main() { log.println("call main") }
我使用以下命令创建了库:
go build -buildmode=c-shared -o lib.so lib.go
为了使用该库,我编写了这段 golang 代码。
package main /* #include#include #include static void callfromlib() { void (*fn)(); void *h = dlopen("lib.so", rtld_lazy); if (!h) { fprintf(stderr, "error: %s\n", dlerror()); return; } *(void**)(&fn) = dlsym(h, "runlib"); if (!fn) { fprintf(stderr, "error: %s\n", dlerror()); dlclose(h); return; } fn(); dlclose(h); } */ import "c" func main() { c.callfromlib() }
如果我运行最后一个代码,它会抛出此错误(go run call.go
):
fatal error: bad sweepgen in refill goroutine 1 [running, locked to thread]: runtime.throw({0xb990782, 0xc000042a38}) /usr/local/go/src/runtime/panic.go:1198 +0x71 fp=0xc000042a18 sp=0xc0000429e8 pc=0xb930691 runtime.(*mcache).refill(0x41215b8, 0x2) /usr/local/go/src/runtime/mcache.go:156 +0x24e fp=0xc000042a68 sp=0xc000042a18 pc=0xb91434e runtime.(*mcache).nextfree(0x41215b8, 0x2) /usr/local/go/src/runtime/malloc.go:880 +0x85 fp=0xc000042ab0 sp=0xc000042a68 pc=0xb90ba85 runtime.mallocgc(0x8, 0xb9ba300, 0x1) /usr/local/go/src/runtime/malloc.go:1071 +0x4e8 fp=0xc000042b30 sp=0xc000042ab0 pc=0xb90c108 runtime.growslice(0xb9ba300, {0x0, 0x41a1910, 0x2}, 0xc00009c000) /usr/local/go/src/runtime/slice.go:267 +0x4ea fp=0xc000042b98 sp=0xc000042b30 pc=0xb94586a sync.(*pool).pinslow(0xba394e0) /usr/local/go/src/sync/pool.go:223 +0x105 fp=0xc000042c30 sp=0xc000042b98 pc=0xb960ec5 sync.(*pool).pin(0xba394e0) /usr/local/go/src/sync/pool.go:206 +0x4e fp=0xc000042c48 sp=0xc000042c30 pc=0xb960d8e sync.(*pool).get(0xba394e0) /usr/local/go/src/sync/pool.go:128 +0x25 fp=0xc000042c80 sp=0xc000042c48 pc=0xb960ac5 fmt.newprinter() /usr/local/go/src/fmt/print.go:137 +0x25 fp=0xc000042ca8 sp=0xc000042c80 pc=0xb985d45 fmt.sprintln({0xc000042d38, 0x1, 0x1}) /usr/local/go/src/fmt/print.go:280 +0x28 fp=0xc000042cf0 sp=0xc000042ca8 pc=0xb986008 log.println({0xc000042d38, 0x24, 0x0}) /usr/local/go/src/log/log.go:329 +0x1e fp=0xc000042d20 sp=0xc000042cf0 pc=0xb98cd5e main.runlib(...) /users/.../demo/lib.go:8 _cgoexp_6b951f94a90e_runlib(0xc000042d90) _cgo_gotypes.go:36 +0x45 fp=0xc000042d58 sp=0xc000042d20 pc=0xb98cf85 runtime.cgocallbackg1(0xb98cf40, 0xc000042e60, 0x0) /usr/local/go/src/runtime/cgocall.go:306 +0x29a fp=0xc000042e28 sp=0xc000042d58 pc=0xb903d1a runtime.cgocallbackg(0xc0000001a0, 0x300000002, 0xc0000001a0) /usr/local/go/src/runtime/cgocall.go:232 +0x109 fp=0xc000042eb8 sp=0xc000042e28 pc=0xb9039e9 runtime.cgocallbackg(0xb98cf40, 0x7ffeefbff737, 0x0):1 +0x2f fp=0xc000042ee0 sp=0xc000042eb8 pc=0xb95e32f runtime: unexpected return pc for runtime.cgocallback called from 0x4053e00 stack: frame={sp:0xc000042ee0, fp:0xc000042f08} stack=[0xc000042000,0xc000043000) 0x000000c000042de0: 0x000000c000042d9d 0x000000c000042e18 0x000000c000042df0: 0x000000000b95843b 0x000000c0000001a0 0x000000c000042e00: 0x000000c000042dd8 0x0000000000000000 0x000000c000042e10: 0x000000000b9c3dc0 0x000000c000042ea8 0x000000c000042e20: 0x000000000b9039e9 0x000000000b98cf40 <_cgoexp_6b951f94a90e_runlib+0x0000000000000000> ... <_cgoexp_6b951f94a90e_runlib+0x0000000000000000> 0x000000c000042ea0: 0x00007ffeefbff737 0x000000c000042ed0 0x000000c000042eb0: 0x000000000b95e32f 0x000000c0000001a0 ... 0x000000c000042ee0: <0x000000000b98cf40 <_cgoexp_6b951f94a90e_runlib+0x0000000000000000> 0x00007ffeefbff737 ... runtime.cgocallback(0x4004165, 0x4058340, 0xc000042f70) /usr/local/go/src/runtime/asm_amd64.s:915 +0xb4 fp=0xc000042f08 sp=0xc000042ee0 pc=0xb95c134 goroutine 1 [runnable, locked to thread]: unicode.init() /usr/local/go/src/unicode/tables.go:9 +0x79 exit status 2
但是如果我使用 python 一切正常!
>>> import ctypes >>> lib = ctypes.cdll.loadlibrary("lib.so") >>> 2022/03/02 01:08:17 call init >>> lib.runlib() 2022/03/02 01:08:22 call runlib 0 >>>
信息
操作系统:macos big sur 11.6.3 (20g415)
>>> clang --version apple clang version 13.0.0 (clang-1300.0.29.30)
>>> nm lib.so| grep runlib 000000000008cfa0 t _runlib 000000000008cf40 t __cgoexp_6b951f94a90e_runlib 00000000000c4200 s __cgoexp_6b951f94a90e_runlib.stkobj
>>> go env GO111MODULE="on" GOARCH="amd64" GOBIN="" GOCACHE="~/Library/Caches/go-build" GOENV="~/Library/Application Support/go/env" GOEXE="" GOEXPERIMENT="" GOFLAGS="" GOHOSTARCH="amd64" GOHOSTOS="darwin" GOINSECURE="" GOMODCACHE="~/go/pkg/mod" GONOPROXY="" GONOSUMDB="" GOOS="darwin" GOPATH="~/go" GOPRIVATE="" GOPROXY="https://proxy.golang.org,direct" GOROOT="/usr/local/go" GOSUMDB="sum.golang.org" GOTMPDIR="" GOTOOLDIR="/usr/local/go/pkg/tool/darwin_amd64" GOVCS="" GOVERSION="go1.17" GCCGO="gccgo" AR="ar" CC="clang" CXX="clang++" CGO_ENABLED="1" GOMOD="~/projects/go/.../go.mod" CGO_CFLAGS="-g -O2" CGO_CPPFLAGS="" CGO_CXXFLAGS="-g -O2" CGO_FFLAGS="-g -O2" CGO_LDFLAGS="-g -O2" PKG_CONFIG="pkg-config" GOGCCFLAGS="-fPIC -arch x86_64 -m64 -pthread -fno-caret-diagnostics -Qunused-arguments -fmessage-length=0 -fdebug-prefix-map=/var/folders/wl/9rtfdx8x7jvgyn6t8hpq7yh00000gn/T/go-build401121298=/tmp/go-build -gno-record-gcc-switches -fno-common"
我用谷歌搜索并找到了这篇文章。但文章中并没有golang。
https://medium.com/learning-the-go-programming-language/calling-go-functions-from-other-languages-4c7d8bcc69bf
正确答案
一位 go 贡献者写道,一个镜像中不可能有两个 go 运行时。该评论指的是 windows,但似乎也适用于 macos,因为我可以可靠地重现 op 提到的崩溃:
参见https://github.com/golang/go/issues/50304#issuecomment-999302888
但是,由于 op 使用 macos,因此还有一种替代方法:使用插件,请参阅 https://pkg.go.dev/plugin:
创建库的命令将更改如下:
go build -buildmode=plugin -o lib.so lib.go
因此,您对库中定义的函数的调用在 go 中将如下所示:
package main import ( "plugin" ) func main() { plug, err := plugin.open("lib.so") if err != nil { panic(err) } runlib, err := plug.lookup("runlib") if err != nil { panic(err) } runlib.(func())() }
使用此程序进行的测试会生成以下日志输出:
stephan@mac golang-c-lib-in-go % go run call.go 2022/10/29 23:23:01 Call init 2022/10/29 23:23:01 Call RunLib
如果该库还用于从 c 和/或 python 调用,那么一种可能性是以两种变体创建该库:一种是从 c 和/或 python 调用,如问题所示,以及插件变体中的一个,然后可以由 go 程序使用。
在您的示例中,您不需要对 lib.go
进行任何更改,唯一的区别是创建库的命令行。
最后,来自 documentation 的重要提示:
今天带大家了解了的相关知识,希望对你有所帮助;关于Golang的技术知识我们会一点点深入介绍,欢迎大家关注golang学习网公众号,一起学习编程~

- 上一篇
- 为何将记录保存在GoLang中的标准输入?

- 下一篇
- Jenkins 搭配 SonarQube 分析 Go 代码的操作方法
-
- Golang · Go问答 | 1年前 |
- 在读取缓冲通道中的内容之前退出
- 139浏览 收藏
-
- Golang · Go问答 | 1年前 |
- 戈兰岛的全球 GOPRIVATE 设置
- 204浏览 收藏
-
- Golang · Go问答 | 1年前 |
- 如何将结构作为参数传递给 xml-rpc
- 325浏览 收藏
-
- Golang · Go问答 | 1年前 |
- 如何用golang获得小数点以下两位长度?
- 477浏览 收藏
-
- Golang · Go问答 | 1年前 |
- 如何通过 client-go 和 golang 检索 Kubernetes 指标
- 486浏览 收藏
-
- Golang · Go问答 | 1年前 |
- 将多个“参数”映射到单个可变参数的习惯用法
- 439浏览 收藏
-
- Golang · Go问答 | 1年前 |
- 将 HTTP 响应正文写入文件后出现 EOF 错误
- 357浏览 收藏
-
- Golang · Go问答 | 1年前 |
- 结构中映射的匿名列表的“复合文字中缺少类型”
- 352浏览 收藏
-
- Golang · Go问答 | 1年前 |
- NATS Jetstream 的性能
- 101浏览 收藏
-
- Golang · Go问答 | 1年前 |
- 如何将复杂的字符串输入转换为mapstring?
- 440浏览 收藏
-
- Golang · Go问答 | 1年前 |
- 相当于GoLang中Java将Object作为方法参数传递
- 212浏览 收藏
-
- Golang · Go问答 | 1年前 |
- 如何确保所有 goroutine 在没有 time.Sleep 的情况下终止?
- 143浏览 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 542次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 508次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 497次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 484次学习
-
- 可图AI图片生成
- 探索快手旗下可灵AI2.0发布的可图AI2.0图像生成大模型,体验从文本生成图像、图像编辑到风格转绘的全链路创作。了解其技术突破、功能创新及在广告、影视、非遗等领域的应用,领先于Midjourney、DALL-E等竞品。
- 14次使用
-
- MeowTalk喵说
- MeowTalk喵说是一款由Akvelon公司开发的AI应用,通过分析猫咪的叫声,帮助主人理解猫咪的需求和情感。支持iOS和Android平台,提供个性化翻译、情感互动、趣味对话等功能,增进人猫之间的情感联系。
- 14次使用
-
- Traini
- SEO摘要Traini是一家专注于宠物健康教育的创新科技公司,利用先进的人工智能技术,提供宠物行为解读、个性化训练计划、在线课程、医疗辅助和个性化服务推荐等多功能服务。通过PEBI系统,Traini能够精准识别宠物狗的12种情绪状态,推动宠物与人类的智能互动,提升宠物生活质量。
- 17次使用
-
- 可图AI 2.0图片生成
- 可图AI 2.0 是快手旗下的新一代图像生成大模型,支持文本生成图像、图像编辑、风格转绘等全链路创作需求。凭借DiT架构和MVL交互体系,提升了复杂语义理解和多模态交互能力,适用于广告、影视、非遗等领域,助力创作者高效创作。
- 19次使用
-
- 毕业宝AIGC检测
- 毕业宝AIGC检测是“毕业宝”平台的AI生成内容检测工具,专为学术场景设计,帮助用户初步判断文本的原创性和AI参与度。通过与知网、维普数据库联动,提供全面检测结果,适用于学生、研究者、教育工作者及内容创作者。
- 32次使用
-
- GoLand调式动态执行代码
- 2023-01-13 502浏览
-
- 用Nginx反向代理部署go写的网站。
- 2023-01-17 502浏览
-
- Golang取得代码运行时间的问题
- 2023-02-24 501浏览
-
- 请问 go 代码如何实现在代码改动后不需要Ctrl+c,然后重新 go run *.go 文件?
- 2023-01-08 501浏览
-
- 如何从同一个 io.Reader 读取多次
- 2023-04-11 501浏览