gpt4 book ai didi

Golang : multiple definition of CGO ported package

转载 作者:行者123 更新时间:2023-12-03 10:01:51 27 4
gpt4 key购买 nike

我有 2 个项目,第一个,名称为 A ,有一个子模块a导入 sqlite3(github.com/mattn/go-sqlite3)。另一个B项目导入A的子模块 a , 在另一个子模块 b ,它也导入相同的sqlite3。

两个AB把进口放在vendor下目录(由 govendor 管理)。我的 Golang 版本是 go version go1.12 linux/amd64 .

构建时 B ( go build main.go ),抛出以下错误(太多,其中一部分):

 /usr/local/go/pkg/tool/linux_amd64/link: running gcc failed: exit status 1
/tmp/go-link-281256755/000029.o: In function `callbackTrampoline':
/tmp/go-build/_cgo_export.c:25: multiple definition of `callbackTrampoline'
/tmp/go-link-281256755/000005.o:/tmp/go-build/_cgo_export.c:25: first defined here
/tmp/go-link-281256755/000029.o: In function `stepTrampoline':
...
/home/xxx/go/src/gitlab.xxxxxxxxx.com/xxxxxxxxx-tools/A/vendor/github.com/mattn/go-sqlite3/sqlite3.go:129: multiple definition of `_sqlite3_result_text'
/tmp/go-link-281256755/000009.o:/home/xxx/go/src/gitlab.xxxxxxxxx.com/xxxxxxxxx-tools/A/vendor/github.com/mattn/go-sqlite3/sqlite3.go:129: first defined here
/tmp/go-link-281256755/000033.o: In function `_sqlite3_result_blob':
...

但建筑 A效果很好。为了测试错误,我开始关注演示,同样使用 vendor。由 govendor 发起,并构建好。
 package main

import (
"database/sql"
"fmt"

"gitlab.xxxxxxxxx.com/xxxxxxxxxxxxxxx/A/a"

_ "github.com/mattn/go-sqlite3"
)

func main() {
fmt.Println(a.ModuleVariable) // use submodule `a` just like B is doing

_, _ = sql.Open(`sqlite3`, `test.db`) // use sqlite too
}

我认为编译器首先编译 Asqlite3 , 对象在 /tmp/go-link-281256755/000005.o 下创建(但构建后没有这个目录),然后编译 B的导入 sqlite3并且还创建了一个包含同名函数的对象,然后编译器找到2个同名符号,链接失败。

如何解决这些情况?是否有任何 golang env 设置来避免这些?

在我删除 vendor 下的 sqlite3 包之后两个 AB , 他们都使用 ~/go/src/github.com/mattn/go-sqlite3/ 下的 sqlite3 ,他们都建立好。但由于项目 A,我不能这样做的部署平台,我必须将所有依赖项放在 vendor 下,是否有其他选项可以对同一个包使用多个导入?

最佳答案

对于cgo的链接错误“...的多重定义”的问题,(解决方法)解决方案取决于链接的 C 代码的性质:

  • 如果两个 Go 包链接到 相同的 C 代码(库) ,你应该通过选项 --allow-multiple-definition到链接器(见 ld man page ),或者通过命令选项go build --ldflags '-extldflags "-Wl,--allow-multiple-definition"' ,
    或通过#cgo链接到 C 代码的包的 Go 源代码中的指令:
    //#cgo LDFLAGS: -Wl,--allow-multiple-definition
    import "C"
  • 如果两个 Go 包链接到包含 的不同 C 代码一些具有相同名称的函数和变量 ,你应该重构那些 C 代码:
  • 确保输入关键字static仅在该 C 对象内使用的所有声明(不打算链接到 Go 或其他 C 对象)。
  • 找到一些方法来进行名称修改或将那些重复的标识符放入不同的 namespace (如在 C++ 中)。最好是cgo支持一些使用 Go 包名进行自动命名空间的机制,但直到现在(2020 年)你必须自己做。 C 预处理器的“标记粘贴”运算符 ##可能有助于此命名空间任务。例如。
    //File: my_package1.h
    #define NS(id) my_package1_ ## id
    void NS(my_function1)(int);
    void NS(my_function2)(float);
    char NS(my_shared_var);

  • 如果您有任何 C函数定义在 Go 源代码中,如 this question ,您必须将这些定义移动到同一包文件夹下的单独 C 源文件中,只在 Go 源中保留声明。
  • 关于Golang : multiple definition of CGO ported package,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/56318343/

    27 4 0
    Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
    广告合作:1813099741@qq.com 6ren.com