为了账号安全,请及时绑定邮箱和手机立即绑定

在 Go 程序中使用 cgo 共享库

在 Go 程序中使用 cgo 共享库

Go
POPMUISE 2022-10-31 15:35:25
试图测试cgo,所以我写了以下内容://go:build lib// +build libpackage mainimport "C"import "fmt"//export HelloWorldfunc HelloWorld() {    fmt.Printf("hello world")}func main() {}// go build -tags lib -buildmode=c-shared -o golib.a lib.go并将其编译为:$ go build -tags lib -buildmode=c-shared -o golib.a lib.go尝试在另一个代码中使用生成的共享库://go:build app// +build apppackage main// #cgo CFLAGS: -g -Wall// #include <stdlib.h>// #include "golib.h"import "C"func main() {    C.HelloWorld()}// go run main.go但我收到以下错误:# command-line-argumentsUndefined symbols for architecture x86_64:  "_HelloWorld", referenced from:      __cgo_a844f0d618a1_Cfunc_HelloWorld in _x002.o     (maybe you meant: __cgo_a844f0d618a1_Cfunc_HelloWorld)ld: symbol(s) not found for architecture x86_64clang: error: linker command failed with exit code 1 (use -v to see invocation)# command-line-argumentscgo-gcc-prolog:47:33: warning: unused variable '_cgo_a' [-Wunused-variable]此外,我在 mac 上使用 VS 代码收到以下错误:go list failed to return CompiledGoFiles. This may indicate failure to perform cgo processing; try building at the command line. See https://golang.org/issue/38990在此处输入图像描述
查看完整描述

1 回答

?
慕桂英3389331

TA贡献2036条经验 获得超8个赞

你不能在 Go 程序中使用 cgo 共享库,因为你不能在同一个进程中有多个 Go 运行时。


尝试这样做会给出错误:


# command-line-arguments

cgo-gcc-prolog:67:33: warning: unused variable '_cgo_a' [-Wunused-variable]

fatal error: unexpected signal during runtime execution

[signal SIGSEGV: segmentation violation code=0x1 addr=0x0 pc=0x43fd5e2]


goroutine 1 [running, locked to thread]:

runtime.throw({0x40a875b?, 0x1c00011b800?})

        /usr/local/go/src/runtime/panic.go:992 +0x71 fp=0x1c00004a960 sp=0x1c00004a930 pc=0x402f6d1

runtime: unexpected return pc for runtime.sigpanic called from 0x43fd5e2

stack: frame={sp:0x1c00004a960, fp:0x1c00004a9b0} stack=[0x1c00004a000,0x1c00004b000)

....

0x000001c00004aaa0:  0x0000000000000000  0x0000000000000000 

runtime.sigpanic()

        /usr/local/go/src/runtime/signal_unix.go:781 +0x3a9 fp=0x1c00004a9b0 sp=0x1c00004a960 pc=0x4043449

exit status 2

要从 go 调用 c 代码,有 3 种可能的方法:


内联编码,如:

package main


//#include  <stdio.h>

//int Add(int a, int b){

//    printf("Welcome from external C function\n");

//    return a + b;

//}

import "C"

import "fmt"


func main() {

    fmt.Println(C.Add(5, 2))

}

静态链接,您同时拥有.c和.h文件,如:

// lib.c

#include  <stdio.h>

int Add(int a, int b){

    printf("Welcome from external C function\n");

    return a + b;

}


// libadd.h

int Add(int, int);

go 文件是:


// main.go

package main


// #include "libadd.h"

import "C"

import "fmt"


func main() {

    x := C.Add(1, 2)

    fmt.Println(x)


}

我们必须将文件运行为go run .或go run github.io/xxx // go run main.go 将不起作用,因为它只会考虑 main.go,而不考虑 C 文件


动态链接,您将上述c文件编译为clang -shared -fpic -Wall -g lib.c -o libadd.so并将 go 文件作为:

// main.go

package main


//#cgo CFLAGS: -g -Wall

//#cgo LDFLAGS: -L. -ladd

//#include "libadd.h"

import "C"

import "fmt"


func main() {

    x := C.Add(1, 2)

    fmt.Println(x)

}

这里你可以使用go run main.go通过硬代码连接的库,分发二进制文件,共享库loadadd.so需要与相同的二进制文件一起分发并且存在于同一个文件夹中。


我在这里为每个案例上传了单独的文件夹。


奖励 要使用 ac 程序在 go 中调用生成的共享库,我们可以使用以下命令:


// main.c

#include <stdio.h>

#include "libadd.h" // calling C file

#include "libgo.h" // calling shared library generated by GO


int main()

{

    HelloWorld();

    int x = Add(1, 2);

    printf("%d",x);

    return 0;

}

将文件编译为:


clang -o main -L. -ladd -lgo main.c   // -ladd => -l (library) is libadd


查看完整回答
反对 回复 2022-10-31
  • 1 回答
  • 0 关注
  • 389 浏览
慕课专栏
更多

添加回答

举报

0/150
提交
取消
微信客服

购课补贴
联系客服咨询优惠详情

帮助反馈 APP下载

慕课网APP
您的移动学习伙伴

公众号

扫描二维码
关注慕课网微信公众号