登录
首页 >  Golang >  Go问答

在现有的 C 项目上使用 Go

来源:Golang技术栈

时间:2023-03-22 14:38:41 199浏览 收藏

今日不肯埋头,明日何以抬头!每日一句努力自己的话哈哈~哈喽,今天我将给大家带来一篇《在现有的 C 项目上使用 Go》,主要内容是讲解golang等等,感兴趣的朋友可以收藏或者有更好的建议在评论提出,我都会认真看的!大家一起进步,一起学习!

问题内容

我有一个完全用 C 语言编写的程序,其中使用了多个目标(.o)文件。这些文件都打包在一个存档文件(.a)中,而该存档文件又在程序主(.c)文件的编译时使用。

我想用 Go 为这个项目写一个新文件。我的想法是编写这个.go文件,然后(.o)从中创建一个目标文件。之后,我想把这个目标文件放在已经提到的存档(.a)文件中。

这基本上意味着 我想从 C 程序中调用 Go 函数 。我读过[这个问题](https://stackoverflow.com/questions/6125683/call-go-functions- from-c),虽然它向我展示了我想要的东西可以通过 GCCGO 实现,但如何做到这一点并不是 100% 清楚。

即使是最基本的测试,我在链接阶段也会出错。更具体地说,这是这样一个基本示例之一:


打印字符串.go

package main

import
(
    "fmt"
)

func PrintString(buff string) int {
    fmt.Printf(buff)
    return 1
}

c_caller.c

#define _GNU_SOURCE
#include 

extern int PrintString(char*) __asm__ ("print.main.PrintString");

int main() {
    char *string_to_pass= NULL;
    asprintf(&string_to_pass, "This is a test.");

    int result= PrintString(string_to_pass);
    if(result) {printf("Everything went as expected!\n");}
    else       {printf("Uh oh, something went wrong!\n");}

    return result;
}

编译

为了编译 Go 文件,我使用了以下命令:

gccgo -c printString.go -o printString.o -fgo-prefix=print -Wall -Werror -march=native

为了编译整个东西,我使用了这个命令:

gccgo -o main c_caller.c printString.o -Wall -Werror -march=native

我收到的返回消息是:

/usr/lib64/libgo.so.4.0.0: undefined reference to `main.main'
/usr/lib64/libgo.so.4.0.0: undefined reference to `__go_init_main'
collect2: error: ld returned 1 exit status

这意味着 GCCGO 期望 Go 文件中的 main 函数而不是 C 文件。

在第二个命令上使用--static- libgo,-static-Wl,-R,/path/to/libgo.so's_folder选项会产生不同的结果:

/usr/bin/ld: cannot find -lgo
collect2: error: ld returned 1 exit status

这没有任何意义,因为我有 LD_LIBRARY_PATH 环境变量正确指向 libgo.so 的文件夹。


我意识到我可能在这里做错了什么,但我就是看不到那是什么。几乎没有 GCCGO 及其与 C 交互的示例,我能找到的唯一参考是这个页面,我个人觉得这还不够。

我恳请您就此事提供一些建议,并感谢您抽出宝贵时间。:)

正确答案

这可能不是您想要的,但在今年 8 月即将推出的 Go 1.5 中,您将能够使用 go 工具构建与 C 兼容的库。所以在你的这个_main.c

#include 

int main()
{
    char *string_to_pass = NULL;
    if (asprintf(&string_to_pass, "This is a test.") 

这在你的main.go

package main

import "C"
import "fmt"

//export PrintString
func PrintString(cs *C.char) {
    s := C.GoString(cs)
    fmt.Println(s)
}

func main() {}

对于静态库,您可以这样做:

go build -buildmode c-archive -o mygopkg.a
gcc -o main _main.c mygopkg.a -lpthread

对于共享库:

go build -buildmode c-shared -o mygopkg.so
LD_RUN_PATH=$(pwd) gcc -o main _main.c mygopkg.so -lpthread

LD_RUN_PATH这里是为了让链接器在您正在构建的同一目录中查找共享库。)

有关更多信息,请参阅[Go 执行模式设计文档](https://docs.google.com/document/d/1nr- TQHw_er6GOQRsF6T43GGhFDelrAP0NqSS_00RgZQ/edit)。

今天关于《在现有的 C 项目上使用 Go》的内容就介绍到这里了,是不是学起来一目了然!想要了解更多关于golang的内容请关注golang学习网公众号!

声明:本文转载于:Golang技术栈 如有侵犯,请联系study_golang@163.com删除
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>
评论列表