在很多场景下,在Go的程序中需要调用c函数或者是用c编写的库(底层驱动,算法等,不想用Go语言再去造一遍轮子,复用现有的c库)。
那么该如何调用呢?Go可是更好的C语言啊,当然提供了和c语言交互的功能,称为Cgo
!Cgo
封装了#cgo
伪c文法,参数CFLAGS
用来传入编译选项,LDFLAGS
来传入链接选项。这个用来调用非c标准的第三方c库。
1)先从最简单的写起吧,Go代码直接调用c函数,下面的示例中在代码注释块调用了标准的c库,并写了一个c函数(本例只是简单打印了一句话,在该注释块中可以写任意合法的c代码),在Go代码部分直接调用该c函数hi()
package main
import "fmt"
/*
#include <stdio.h>
void hi() {
printf("hello world!\n");
}
*/
import "C" //这里可看作封装的伪包C, 这条语句要紧挨着上面的注释块,不可在它俩之间间隔空行!
func main() {
C.hi()
fmt.Println("Hi, vim-go")
}
运行结果:
root@slave2:/home/cgo# go run main.go
hello world!
Hi, vim-go
好,我可以在Go代码中写c代码了,那么我该如何在Go中直接调用已经编译好的第三方c库呢?用Cgo
!
2)本例示范在Go代码中调用非标准的c的第三方动态库
c文件
/*
* hi.c
* created on: July 1, 2017
* author: mark
*/
#include <stdio.h>
void hi() {
printf("Hello Cgo!\n");
}
h文件
void hi();
编译成动态库.so
root@slave2:/home/cgo# gcc -c -fPIC -o hi.o hi.c
root@slave2:/home/cgo# gcc -shared -o libhi.so hi.o
Go文件
package main
import "fmt"
/*
#cgo CFLAGS: -I./
#cgo LDFLAGS: -L./ -lhi
#include "hi.h" //非标准c头文件,所以用引号
*/
import "C"
func main() {
C.hi()
fmt.Println("Hi, vim-go")
}
重点来了(敲黑板):CFLAGS
中的-I
(大写的i) 参数表示.h
头文件所在的路径LDFLAGS
中的-L
(大写) 表示.so文件所在的路径 -l
(小写的L) 表示指定该路径下的库名称,比如要使用libhi.so
,则只需用-lhi
(省略了libhi.so
中的lib
和.so
字符,关于这些字符所代表的具体含义请自行google)表示。
运行结果:
root@slave2:/home/cgo# go run main.go
Hello Cgo!
Hi, vim-go
更深入一些,
1)头文件路径和库文件路径写死的话,一旦第三方库的安装路径变化了,Golang的代码也要跟着变化,就会很麻烦。这时可以使用cgo
命令中使用pk-config
,具体请参考这篇博文:Golang使用pkg-config自动获取头文件和链接库的方法
2)当在Go中使用了以上的方法后,就要求主机(或者云服务器)上必须有相应的.so
文件,如果不存在就会链接报错,导致程序退出。
若.so
是一些不必要的第三方驱动库(可有可无),那就麻烦了,你不能为了跑这个程序,把每台主机都装上那个不必要的第三方库吧。有没有一种方法可以在Go程序运行时才调用这些.so
库呢,如果不存在忽略就好(就不启用那个库提供的功能了,而不是链接报错直接异常退出)?当然有!敬请期待一下一篇。:)
有疑问加站长微信联系(非本文作者)