0

0

透过一个编译报错,总结两个Go程序编译的重要知识

蓮花仙者

蓮花仙者

发布时间:2025-10-02 10:21:20

|

679人浏览过

|

来源于php中文网

原创

最近调研了一下某个做 apm 的厂商的 go 探针程序,说是引入一个包,全程不用再修改其他代码就能在项目里引入探针。没想到在刚引入包试着构建了一下就翻车了。

代码语言:javascript代码运行次数:0运行复制
main.go:10:2: build constraints exclude all Go files in /xxx/github.com/xxx/agnet/xxxx

编译器编译的时候直接排除了某个包下的所有文件,啥意思呢?就是这个包下没有能在当前构建环境下构建的 Go 文件。猜测应该是这个包源码的构建标签上声明了不允许在Mac 环境下构建。打开源码看了看,确实是,所有文件的构建标签都是这么声明的。

代码语言:javascript代码运行次数:0运行复制
// +build linux// +build amd64

这个叫做条件编译,或者约束编译。那想在Mac下编译linux上才能运行的执行文件该怎么办呢?Go 里边还支持一个特性叫做交叉编译,就是干跨平台编译这个事儿的。具体怎么用呢,比如这个例子里是需要在Mac环境下编译能在Linux系统amd64架构下运行的执行文件,就得酱婶进行编译:

代码语言:javascript代码运行次数:0运行复制
CGO_ENABLED=0  GOOS=linux  GOARCH=amd64  go build main.go

不过我后来想研究下为啥不让在 Mac 上编译,看了看这个包的探针是用CGO实现的调用了linux系统下的一个C语言实现的工具命令。看到这我已经不想继续研究这个包了,那么为了让此篇文章水的不那么明显:),接下来咱们就把Go语言的交叉编译和条件编译这两个知识点再复习一遍吧。

交叉编译

交叉编译是用来在一个平台上生成另一个平台的可执行程序。Go 的命令集是原生支持交叉编译的,使用方法也很简单,比如上面已经演示过的

代码语言:javascript代码运行次数:0运行复制
CGO_ENABLED=0  GOOS=linux  GOARCH=amd64  go build main.go
参数说明CGO_ENABLED : CGO 表示golang中的工具,CGO_ENABLED 表示CGO禁用,交叉编译中不能使用CGO的GOOS: 目标平台mac 对应 darwinlinux 对应 linuxwindows 对应 windowsGOARCH :目标平台的体系架构【386,amd64,arm】, 目前市面上的个人电脑一般都是amd64架构的386 也称 x86 对应 32位操作系统amd64 也称 x64 对应 64位操作系统arm 这种架构一般用于嵌入式开发。比如 Android , IOS , Win mobile , TIZEN 等

了解完这几个参数后,我们在看下Mac、Linux、Windows这三个平台上执行交叉编译的例子,Windows的因为家境贫寒,条件不允许我没有试过,命令网上找的,如果有错误还请同学们在评论里留言帮我改正一下。

Mac 下编译, Linux 或者 Windows 的可执行程序代码语言:javascript代码运行次数:0运行复制
# linux可执行程序CGO_ENABLED=0  GOOS=linux  GOARCH=amd64  go build main.go# Windows 可执行程序CGO_ENABLED=0 GOOS=windows  GOARCH=amd64  go  build  main.go
Linux 下编译 , Mac 或者 Windows 下去执行代码语言:javascript代码运行次数:0运行复制
# Mac 平台可执行程序CGO_ENABLED=0 GOOS=darwin  GOARCH=amd64  go build main.go# Windows可执行程序CGO_ENABLED=0 GOOS=windows  GOARCH=amd64  go build main.go
Windows 下执行 , Mac 或 Linux 下去执行

需要写一个批处理程序,在里面去设置,因为windows 下的 terminal 不支持shell , 这跟 Mac 和 Linux下的有点不同

代码语言:javascript代码运行次数:0运行复制
# Mac 下执行SET  CGO_ENABLED=0SET GOOS=darwinSET GOARCH=amd64go build main.go# Linux 去执行SET CGO_ENABLED=0SET GOOS=linuxSET GOARCH=amd64go build main.go
条件编译

交叉编译只是为了能在一个平台上编译出其他平台可运行的程序,Go 作为一个跨平台的语言,它提供的类库势必也是跨平台的,比如说程序的系统调用相关的功能,能根据所处环境选择对应的源码进行编译。让编译器只对满足条件的代码进行编译,将不满足条件的代码舍弃,这就是另外一个概念叫---条件编译。

在 Go 中,也称之为Build Constraints 编译约束,添加编译约束的方式有2种分别:

编译标签(build tag)文件后缀编译标签

编译标签是一种通过在源码文件顶部添加注释,来决定文件是否参与编译的约束方式。其格式如下:

代码语言:javascript代码运行次数:0运行复制
// +build 

注意:// +build的下一行必须是空行,否则会被解析为包注释。

代码语言:javascript代码运行次数:0运行复制
// +build linux// main package commentpackage main

tags说明:

以空格分开表示 AND以逗号分开表示 OR!表示 NOT

标签可以指定为以下内容:

InsCode
InsCode

InsCode 是CSDN旗下的一个无需安装的编程、协作和分享社区

下载
操作系统,环境变量中GOOS的值,如:linuxdarwinwindows等等。操作系统的架构,环境变量中GOARCH的值,如:arch64x86i386等等。使用的编译器,gc或者gccgo。是否开启CGO,cgo。golang版本号:比如Go Version 1.1为go1.1,Go Version 1.12版本为go1.12,以此类推。其它自定义标签,通过go build -tags指定的值。

例如,编译条件为(linux AND 386) OR (darwin AND (NOT cgo))

代码语言:javascript代码运行次数:0运行复制
// +build linux,386 darwin,!cgo

另外一个文件可以有多个编译约束,比如条件为(linux OR darwin) AND amd64

代码语言:javascript代码运行次数:0运行复制
// +build linux darwin// +build amd64

也可以使用ignore标签将一个文件从编译中排除。

代码语言:javascript代码运行次数:0运行复制
// +build ignore
文件后缀

除了编译标签,第二种添加编译约束的方法是通过源码文件的文件名实现的,这种方案比构造标签方案更简单。编译器也会根据文件后缀来自动选择编译文件:

代码语言:javascript代码运行次数:0运行复制
$filename_$GOOS.go$filename_$GOARCH.go$filename_$GOOS_$GOARCH.go
$filename: 源文件名称。$GOOS: 表示操作系统,从环境变量中获取。$GOARCH: 表示系统架构,从环境变量中获取。

后缀的顺序记住不要颠倒,后缀中同时出现系统和架构名时,需要保持filename_GOOS_

在 Go 的每个内置库里都有很多以不同系统名结尾的文件。下面是Go的os内置库源代码的部分截图:

透过一个编译报错,总结两个Go程序编译的重要知识

文件后缀添加编译约束

两种添加编译限制的方式该如何选择

构建标签和文件名后缀在功能上是重叠的。比如,一个名为mypkg_linux.go的文件,再包含构建标签// +build linux会显得多余。

通常来说,当只有一个特定平台需要指定时,我们选择文件名后缀的方式。比如:

代码语言:javascript代码运行次数:0运行复制
mypkg_linux.go         // 只在 linux 系统编译mypkg_windows_amd64.go // 只在 windows amd 64位 平台编译

相反,如果你的文件需要指定给多个平台或体系架构使用,或者你需要排除某个特定平台时,我们选择构建标签的方式。比如:

代码语言:javascript代码运行次数:0运行复制
// 在所有类unix平台编译// +build darwin dragonfly freebsd linux netbsd openbsd// 在非Windows平台编译// +build !windows
总结

一个编译器报错,居然水了一篇文章....啊...(咳嗽声)引出来的交叉编译和条件编译(编译约束)这两个非常重要的知识点,其实这两个知识点在很早之前我也写过篇文章,这次相当于从实际遇到问题带出从头开始再分析一遍,希望大家能喜欢。

参考链接http://www.oskip.com/post/golang/golang-build/https://juejin.cn/post/6844903944808824845https://mp.weixin.qq.com/s/Ys8o4arwIFYB6DPCdiGNNQ

相关专题

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

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

395

2023.06.20

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

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

618

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关键字可用于函数的参数中,表示该参数在函数内部不可修改等等。

525

2023.09.20

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

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

640

2023.09.20

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

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

601

2023.09.22

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

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

19

2026.01.20

热门下载

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

精品课程

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

共18课时 | 4.7万人学习

PostgreSQL 教程
PostgreSQL 教程

共48课时 | 7.5万人学习

NumPy 教程
NumPy 教程

共44课时 | 2.9万人学习

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

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