Golang:CGO移植包的多重定义

coa*_*nor 5 linker go cgo

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

并将导入放在 dir 下A(由 管理)。我的 Golang 版本是.Bvendorgovendorgo version go1.12 linux/amd64

在构建Bgo 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':
 ...
Run Code Online (Sandbox Code Playgroud)

但建筑A效果很好。为了测试错误,我开始关注演示,也使用vendorinited bygovendor和 build ok。

 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
 }
Run Code Online (Sandbox Code Playgroud)

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

如何解决这些情况?是否有任何 golang 环境设置可以避免这些问题?


当我删除和vendor下的sqlite3 包后,它们都使用 下的 sqlite3 ,它们都构建正常。但我不能做到这些,由于项目的部署平台,我必须将所有依赖项放在供应商下,是否还有其他选项可以对同一包使用多个导入?AB~/go/src/github.com/mattn/go-sqlite3/A

Har*_*ada 3

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

  1. 如果两个 Go 包链接到相同的 C 代码(库),您应该通过命令选项将选项传递--allow-multiple-definition给链接器(请参阅ld手册页

    go build --ldflags '-extldflags "-Wl,--allow-multiple-definition"',

    或者通过#cgo链接到 C 代码的包的 Go 源代码中的指令:

    //#cgo LDFLAGS: -Wl,--allow-multiple-definition
    import "C"
    
    Run Code Online (Sandbox Code Playgroud)
  2. 如果两个 Go 包链接到不同的 C 代码,其中包含一些具有相同名称的函数和变量,则应该重构这些 C 代码:

    • 确保将关键字放入static仅在该 C 对象内使用的所有声明(不打算链接到 Go 或其他 C 对象)。
    • 找到一些方法来进行名称修改或将这些重复的标识符放入不同的命名空间中(例如在 C++ 中)。如果支持某种机制使用 Go 包名称进行自动命名空间会更好cgo,但直到现在(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);
      
      Run Code Online (Sandbox Code Playgroud)
  3. 如果Go 源代码中有任何C 函数定义(如本问题中所示),则必须将这些定义移至同一包文件夹下的单独 C 源文件中,仅在 Go 源代码中保留声明。