返回介绍

上卷 程序设计

中卷 标准库

下卷 运行时

源码剖析

附录

11.2 CGO

发布于 2024-10-12 19:15:50 字数 1871 浏览 0 评论 0 收藏 0

在 Go 和 C 间相互调用,但是 CGO is not Go! 。除非必要,不建议使用!

  • 编译慢,调试麻烦。
  • 不支持交叉编译。
  • 存在性能问题,以及内存泄漏风险。
  • 需在 C 和 Go 栈间切换。
  • 可能导致线程数量激增。
  • CGO_ENABLED 控制。

直接以注释方式将 C 代码嵌入到 Go 源文件内。

package main

/*
   #include <stdio.h>
   void hello()
   {
       printf("hello, world!\n");
   }
*/
import "C"       // 注意!两者之间不同有空行。

func main() {
    C.hello()
}

独立文件

单独保存在 .c 文件内,以 GCC/GDB 单独调试。

等 C 代码无误后,在 Go 内 #include 头文件。

// hello.h

#ifndef __HELLO_H__
#define __HELLO_H__

void hello();

#endif
// hello.c

#include <stdio.h>
#include "hello.h"

void hello() {
    printf("hello, world!\n");
}

#ifdef __DEBUG__
int main() {
    hello();
    return 0;
}
#endif
$ gcc -g -O0 -D__DEBUG__ -o hello hello.c

$ ./hello
hello, world!

包含头文件,编译器会自动编译(链接) .c.s.go 文件。

package main

/*
    #include "hello.h"
*/
import "C"

func main() {
    C.hello()
}

动态链接

也可将 .c 编译成动态库( .so ),链接到 Go 程序。

$ mkdir ./lib          # 避免在当前目录下被 go 编译器直接编译。
$ mv hello.* ./lib     # 将 hello.h, hello.c 移动到子目录。
$ cd ./lib             # 切换到子目录。

$ gcc -g -O0 -fPIC -shared -o libhello.so hello.c
package main

/*
    #cgo CFLAGS: -I${SRCDIR}/lib
    #cgo LDFLAGS: -L${SRCDIR}/lib -lhello -Wl,-rpath=.:./lib
    #include "hello.h"
*/
import "C"

func main() {
    C.hello()
}
$ go build -o test

$ ldd ./test
libhello.so => ./lib/libhello.so (0x00007fc9fe587000)

静态链接

libc 链接到可执行文件内。

$ go build --ldflags '-extldflags "-static"'

CGO command

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。
列表为空,暂无数据
    我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
    原文