0

0

在Linux环境中使用Go编译静态二进制文件[译]

絕刀狂花

絕刀狂花

发布时间:2025-06-21 12:54:01

|

1016人浏览过

|

来源于php中文网

原创

Part1 引言

Go语言的一个优势是能够生成静态链接的可执行程序。但是,这并不是说默认情况下编译出来的Go可执行程序都是静态链接的。在有些情况下,需要额外的操作才能实现。具体情况取决于操作系统,本文介绍Unix系统下如何达成这一目标。

Part2 示例

下面是用Go语言编写的hello world程序,在linux机器上将其编译成可执行文件。然后检查该可执行文件是静态链接还是动态链接。

代码语言:javascript代码运行次数:0运行复制
package mainimport "fmt"func main() { fmt.Println("hello world")}

编译helloworld可执行文件,操作如下。

在Linux环境中使用Go编译静态二进制文件[译]
在Linux环境中使用Go编译静态二进制文件[译]
Part3 检查可执行程序是动态链接还是静态链接

有多种方法检查可执行程序链接类型,这里介绍三种方法:

1 通过file命令

file命令输出中明确说明helloworld可执行文件为 statically linked类型,即静态链接。

在Linux环境中使用Go编译静态二进制文件[译]
2 通过ldd命令

ldd命令会输出helloworld程序依赖的动态库,由于helloworld非动态链接,所以输出结果为 不是动态可执行程序

在Linux环境中使用Go编译静态二进制文件[译]3 通过nm命令

使用nm命令列举出helloworld中未定义符号(期望在链接运行时通过动态库加载)。输出为空,表明helloworld中没有任何未定义符号。

在Linux环境中使用Go编译静态二进制文件[译]
Part4 DNS和用户组

在Unix机器上,当满足特定条件时,Go语言标准库会借助libc实现DNS和用户组功能。

当cgo启用时(默认情况下,在Unix系统下 CGO_ENABLED 值为1,表示默认开启)。Go语言标准库中的net包会调用C语言库实现DNS查找功能。这意味着Go程序在进行域名解析时,会使用底层操作系统提供的C库函数,如getaddrinfo和getnameinfo。同理,当启用cgo时,Go语言标准库中的os/user包会调用C语言库来查找用户和用户组。

DNS查询代码如下,保存在lookuphost.go文件中。

代码语言:javascript代码运行次数:0运行复制
package mainimport ( "fmt" "net")func main() { fmt.Println(net.LookupHost("go.dev"))}
在Linux环境中使用Go编译静态二进制文件[译]

编译出可执行程序 lookuphost

在Linux环境中使用Go编译静态二进制文件[译]

通过ldd命令可以看出lookuphost可执行程序是一个动态链接,需要在运行时通过ilbc加载共享库。

在Linux环境中使用Go编译静态二进制文件[译]

为啥编译出来的程序是动态链接在官方net包文档中有详细说明。Go语言标准库也提供了纯Go实现的DNS功能(尽管纯Go实现可能缺少一些高级特性),如果我们想使用纯Go实现,而不是依赖于系统的libc,可以在编译的时候设置tags实现。具体操作如下:

在Linux环境中使用Go编译静态二进制文件[译]

此外,我们还可以通过关闭cgo来编译出静态链接程序。在Unix系统中,默认cgo是开启的,查看方法如下。

在Linux环境中使用Go编译静态二进制文件[译]

关闭cgo再次编译lookuphost程序。

在Linux环境中使用Go编译静态二进制文件[译]

查找用户的用户组信息程序如下,代码保存在userlookup.go文件中。

Tana
Tana

“节点式”AI智能笔记工具,支持超级标签。

下载
代码语言:javascript代码运行次数:0运行复制
package mainimport (  "encoding/json"  "log"  "os"  "os/user")func main() {  user, err := user.Lookup("bob")  if err != nil {    log.Fatal(err)  }  je := json.NewEncoder(os.Stdout)  je.Encode(user)}

编译上述代码,然后通过ldd命令看到可执行程序为动态链接。

在Linux环境中使用Go编译静态二进制文件[译]

同上面的DNS程序,我们可以添加编译tags,将其编译为静态链接程序。

在Linux环境中使用Go编译静态二进制文件[译]

此外,我们也可以通过关闭cgo达到同样的效果。

在Linux环境中使用Go编译静态二进制文件[译]
Part5 将C代码链接到Go二进制文件

Go语言支持通过cgo调用C语言中的函数接口(FFI),下面通过一个具体例子说明,下述代码保存在cstdio.go文件中。

代码语言:javascript代码运行次数:0运行复制
package main//#include // void helloworld(){// printf("hello,world from C\n");//}import "C"func main() { C.helloworld()}

由于使用了cgo,C代码中调用了printf函数,该函数属于libc,即使程序中没有显式调用C运行时,当Go代码通过cgo与C代码交互时,cgo会生成一些“胶水代码”,确保程序能够正常执行。通过ldd命令可以看到上述程序编译后的可执行文件为动态链接类型。

在Linux环境中使用Go编译静态二进制文件[译]

注意:即使我们编写的程序中没有C代码,cgo也可能会涉及。因为我们程序引用的依赖库可能使用了cgo,像常用的go-sqlite3驱动程序就需要cgo,如果我们编写的程序导入了该包,则自然使用了cgo。

这种情况下,通过关闭CGO_ENABLED是无效的。那有什么解决方法吗?请看下一章节内容。

在Linux环境中使用Go编译静态二进制文件[译]
Part6 链接静态libc

如果Go程序包含了C代码,在Unix系统上编译出来的二进制文件是动态链接。具体原因如下:

C代码调用libc(C运行时)在Unix系统上通常使用的libc是glibc推荐的方式是通过动态链接到glibc因此,go build得到的二进制文件是动态链接类型

我们可以换用其他的libc,而不是默认的glibc,比如使用静态链接的libc,这样编译后的可执行文件就是静态的。目前musl就是一个满足我们期望的libc,它是一个轻量级的C标准库,兼容POSIX的API,是许多静态链接应用程序和容器化应用程序的首选。

1 安装musl从官网下载musl源码
在Linux环境中使用Go编译静态二进制文件[译]
解压源码压缩包 musl-1.2.5.tar.gz
在Linux环境中使用Go编译静态二进制文件[译]
进入 musl-1.2.5目录,执行 ./configure --prefix=, MUSLDIR是安装路径,我这里选择的是/usr/local/bin/最后执行 make / make install 进行安装, 安装成功如下
在Linux环境中使用Go编译静态二进制文件[译]
2 编译

下面采用musl对cstdio.go文件进行重行编译,操作如下。CC告诉go build 使用哪个c编译器进行cgo编译。后面的连接器参数设置使用外部连接器。最后执行静态链接。详细信息阅读官方文档

在Linux环境中使用Go编译静态二进制文件[译]

上述编译静态程序的方法同样适用于复杂程序,作者提供了一个use-sqlite.go文件,使用了go-sqlite3包,下面通过两种方式编译对比效果。

代码语言:javascript代码运行次数:0运行复制
// use-sqlite.go package mainimport ( "database/sql" "fmt" "log" _ "github.com/mattn/go-sqlite3")func main() { // Open the database file in /tmp/ db, err := sql.Open("sqlite3", "/tmp/example.db") if err != nil {  log.Fatal(err) } defer db.Close() // Create a table if it doesn't exist createTableSQL := `CREATE TABLE IF NOT EXISTS users (  id INTEGER PRIMARY KEY AUTOINCREMENT,  name TEXT NOT NULL,  age INTEGER NOT NULL );` _, err = db.Exec(createTableSQL) if err != nil {  log.Fatal(err) } // Insert some data insertUserSQL := `INSERT INTO users (name, age) VALUES (?, ?)` _, err = db.Exec(insertUserSQL, "Alice", 30) if err != nil {  log.Fatal(err) } _, err = db.Exec(insertUserSQL, "Bob", 25) if err != nil {  log.Fatal(err) } // Query the data querySQL := `SELECT id, name, age FROM users` rows, err := db.Query(querySQL) if err != nil {  log.Fatal(err) } defer rows.Close() fmt.Println("User data:") for rows.Next() {  var id int  var name string  var age int  err = rows.Scan(&id, &name, &age)  if err != nil {   log.Fatal(err)  }  fmt.Printf("ID: %d, Name: %s, Age: %d\n", id, name, age) } err = rows.Err() if err != nil {  log.Fatal(err) }}
采用默认编译器编译
在Linux环境中使用Go编译静态二进制文件[译]
采用musl编译
在Linux环境中使用Go编译静态二进制文件[译]

采用musl,我们可以在不使用-tags netgo标签,也不用禁用cgo的情况,编译一个静态链接的lookuphost程序。

在Linux环境中使用Go编译静态二进制文件[译]
Part7 使用Zig作为C编译器

Zig 是一种新的系统编程语言,与Go语言类似,它也提供了一系列编译工具即Zig工具链,该工具链包含有Zig编译器、C/C++编译器、链接器和用于静态链接的libc。所以Zig可以用来将Go二进制文件与C代码静态链接。

安装Zig后采用下面的命令编译可执行程序,其中ZIGDIR为Zig的安装目录。相比上一章节的musl-gcc,调用命令会简单一些。

代码语言:javascript代码运行次数:0运行复制
$ CC="$ZIGDIR/zig cc -target x86_64-linux-musl" go build cstdio.go$ CC="$ZIGDIR/zig cc -target x86_64-linux-musl" go build use-sqlite.go
Part8 总结大部分情况下Go语言编译出来的可执行程序是静态,但在某些情况下默认编译出来的是动态链接,我们可以设置build tags或关闭cgo来编译静态链接程序。Go语言中有一个提案,即在 go build 命令中添加一个 -static 标志,表明将程序编译为静态类型,目前还未实现。

相关专题

更多
C语言变量命名
C语言变量命名

c语言变量名规则是:1、变量名以英文字母开头;2、变量名中的字母是区分大小写的;3、变量名不能是关键字;4、变量名中不能包含空格、标点符号和类型说明符。php中文网还提供c语言变量的相关下载、相关课程等内容,供大家免费下载使用。

395

2023.06.20

c语言入门自学零基础
c语言入门自学零基础

C语言是当代人学习及生活中的必备基础知识,应用十分广泛,本专题为大家c语言入门自学零基础的相关文章,以及相关课程,感兴趣的朋友千万不要错过了。

617

2023.07.25

c语言运算符的优先级顺序
c语言运算符的优先级顺序

c语言运算符的优先级顺序是括号运算符 > 一元运算符 > 算术运算符 > 移位运算符 > 关系运算符 > 位运算符 > 逻辑运算符 > 赋值运算符 > 逗号运算符。本专题为大家提供c语言运算符相关的各种文章、以及下载和课程。

354

2023.08.02

c语言数据结构
c语言数据结构

数据结构是指将数据按照一定的方式组织和存储的方法。它是计算机科学中的重要概念,用来描述和解决实际问题中的数据组织和处理问题。数据结构可以分为线性结构和非线性结构。线性结构包括数组、链表、堆栈和队列等,而非线性结构包括树和图等。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

257

2023.08.09

c语言random函数用法
c语言random函数用法

c语言random函数用法:1、random.random,随机生成(0,1)之间的浮点数;2、random.randint,随机生成在范围之内的整数,两个参数分别表示上限和下限;3、random.randrange,在指定范围内,按指定基数递增的集合中获得一个随机数;4、random.choice,从序列中随机抽选一个数;5、random.shuffle,随机排序。

600

2023.09.05

c语言const用法
c语言const用法

const是关键字,可以用于声明常量、函数参数中的const修饰符、const修饰函数返回值、const修饰指针。详细介绍:1、声明常量,const关键字可用于声明常量,常量的值在程序运行期间不可修改,常量可以是基本数据类型,如整数、浮点数、字符等,也可是自定义的数据类型;2、函数参数中的const修饰符,const关键字可用于函数的参数中,表示该参数在函数内部不可修改等等。

524

2023.09.20

c语言get函数的用法
c语言get函数的用法

get函数是一个用于从输入流中获取字符的函数。可以从键盘、文件或其他输入设备中读取字符,并将其存储在指定的变量中。本文介绍了get函数的用法以及一些相关的注意事项。希望这篇文章能够帮助你更好地理解和使用get函数 。

640

2023.09.20

c数组初始化的方法
c数组初始化的方法

c语言数组初始化的方法有直接赋值法、不完全初始化法、省略数组长度法和二维数组初始化法。详细介绍:1、直接赋值法,这种方法可以直接将数组的值进行初始化;2、不完全初始化法,。这种方法可以在一定程度上节省内存空间;3、省略数组长度法,这种方法可以让编译器自动计算数组的长度;4、二维数组初始化法等等。

600

2023.09.22

Java JVM 原理与性能调优实战
Java JVM 原理与性能调优实战

本专题系统讲解 Java 虚拟机(JVM)的核心工作原理与性能调优方法,包括 JVM 内存结构、对象创建与回收流程、垃圾回收器(Serial、CMS、G1、ZGC)对比分析、常见内存泄漏与性能瓶颈排查,以及 JVM 参数调优与监控工具(jstat、jmap、jvisualvm)的实战使用。通过真实案例,帮助学习者掌握 Java 应用在生产环境中的性能分析与优化能力。

3

2026.01.20

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Go 教程
Go 教程

共32课时 | 4万人学习

Go语言实战之 GraphQL
Go语言实战之 GraphQL

共10课时 | 0.8万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号