• Go构建遇到cgo动态库时解决方案


    1. 问题

    1. golang构建程序很简单,当遇到需要调用c库时,如通常使用 net,kafka, sqlite3 程序运行时就会调用当前服务器的 动态库,如果遇到没有库时,通常还需要 下载比如 alpine需要安装sqlite

      apk add --no-cache sqlite-libs sqlite-dev
    2. 通常我们构建时使用CGO_ENABLED=1 就能在构建时将代码需要调用C库用动态连接的形势供代码调用

      CGO_ENABLED=1 go build -ldflags "-s -w" -o perception_node ./cmd/

    3. 但是这里面会有个问题,如果将编译好的 二进制文件移植到其他服务器,但是服务器上面的动态库版本又和构建时的动态库版本不一样, 或者动态库的路径不一样。可能有想到,升级c库,或降低版本。但是一旦升级或降级C库,很可能导致服务器上原来的服务受影响。


    2. 解决

    如何解决:

    1. 使用 golang 的 kafka 库:开发人员需要更改代码以切换使用的 kafka sdk。这可以作为替代方案。
    2. 降低 golang:latest 的 glibc 版本:发行版通常修复 glibc 以编译其他工具链,替换 glibc 是不明智的。虽然有这样的工具yum downgrade glibc*可以帮助解决这个问题。
    3. 更改为旧的 glibc 映像:同样,您无法避免一堆旧的 bash 脚本。
    4. 静态链接 c 依赖项

    综上所述,使用最新的镜像来编译,但是会依赖所有的静态链接,这样一编译完成后就不用担心c库兼容的问题, 如果使用 glibc,则它不是静态可链接的。

    因为 glibc 依赖于支持不同提供程序的 libnss,所以它必须动态链接。

    1. 所以这里替换glibc的唯一方法就是使用musl。librdkafka和 golang 包confluent-kafka-go都支持 musl 构建(构建时指定 –tags musl 即可) alpine 是基于 musl 的发行版,所以这里可以直接用 alpine Linux 构建。

    2. 然后指定外部 ld-staticfor 标志,编译后的二进制文件将完全静态链接。编译过程如下。

      $ docker run -it -v $(pwd):/workspace golang:1.18-alpine
      /go $ cd /workspace/
      /workspace $ sed -i 's/dl-cdn.alpinelinux.org/mirrors.aliyun.com/g' /etc/apk/repositories && \
      > apk add git openssh make build-base alpine-sdk
      fetch https://dl-cdn.alpinelinux.org/alpine/v3.14/main/x86_64/APKINDEX.tar.gz
      fetch https://dl-cdn.alpinelinux.org/alpine/v3.14/community/x86_64/APKINDEX.tar.gz
      (1/37) Installing fakeroot (1.25.3-r3)
      (2/37) Installing openssl (1.1.1l-r0)
      (3/37) Installing libattr (2.5.1-r0)
      (4/37) Installing attr (2.5.1-r0)
      (5/37) Installing libacl (2.2.53-r0)
      (6/37) Installing tar (1.34-r0)
      (7/37) Installing pkgconf (1.7.4-r0)
      ...
      $ export GOPROXY="https://goproxy.cn"
      /workspace $ go build -ldflags "-linkmode external -extldflags '-static'" -tags musl -o perception_node ./cmd/
      /workspace $ ldd perception_node
      /lib/ld-musl-x86_64.so.1: spex: Not a valid dynamic program
    3. windows下使用如下指令

      $ docker run -it -v $(pwd):/workspace golang:1.18-alpine
      /go $ cd /workspace/
      /workspace $ sed -i 's/dl-cdn.alpinelinux.org/mirrors.aliyun.com/g' /etc/apk/repositories
      $ apk add openssh make build-base alpine-sdk mingw-w64-gcc musl-dev gcc build-base
      fetch https://dl-cdn.alpinelinux.org/alpine/v3.14/main/x86_64/APKINDEX.tar.gz
      fetch https://dl-cdn.alpinelinux.org/alpine/v3.14/community/x86_64/APKINDEX.tar.gz
      (1/37) Installing fakeroot (1.25.3-r3)
      (2/37) Installing openssl (1.1.1l-r0)
      (3/37) Installing libattr (2.5.1-r0)
      (4/37) Installing attr (2.5.1-r0)
      (5/37) Installing libacl (2.2.53-r0)
      (6/37) Installing tar (1.34-r0)
      (7/37) Installing pkgconf (1.7.4-r0)
      ...
      $ export GOPROXY="https://goproxy.cn"
      $ CGO_ENABLED=1 GOOS=windows CC=x86_64-w64-mingw32-gcc go build -ldflags "-linkmode external -extldflags '-static'" -tags musl -o seduce_node_agent.exe main.go

    参考

    1.golang动态链接库问题

  • 相关阅读:
    【C语言】字符函数、字符串函数与内存函数
    什么是粘性代理IP
    代码随想录笔记_动态规划_279完全平方数
    本地环境下启动openFaas创建的Java的云函数
    维度灾难 维数灾难 暂记
    numpy学习笔记
    第十二章 Python正则表达式
    基于springboot+vue的药店管理系统
    【HTTP协议——八股文(中)篇】
    Mysql日志管理
  • 原文地址:https://www.cnblogs.com/failymao/p/17189120.html