go 7年前

Golang 本地编译和交叉编译

作者头像 刘宇帅
6821 0

Go 语言的可移植性

Java 平台可移植性是众所都知的,Java 的可移植性依赖于其虚拟机 JVM,Java 实现了对不同平台的 JVM 的支持,那么一份 Java 代码就可以在各个平台上运行。而 Go 语言的可移植性也是依赖于其 runtime,runtime 去对接操作系统层,用户代码在 runtime 中运行,用户代码就不用去关心平台问题。
查看 Go 支持的OS和平台:

> $ go version
go version go1.11 darwin/amd64
 liushuai@liushuaideMacBook-Pro  ~/Documents/goProject/src/qlab/homeguard 
> $ go tool dist list
android/386
android/amd64
android/arm
android/arm64
darwin/386
darwin/amd64
darwin/arm
darwin/arm64
dragonfly/amd64
freebsd/386
freebsd/amd64
freebsd/arm
js/wasm
linux/386
linux/amd64
linux/arm
linux/arm64
linux/mips
linux/mips64
linux/mips64le
linux/mipsle
linux/ppc64
linux/ppc64le
linux/riscv64
linux/s390x
nacl/386
nacl/amd64p32
nacl/arm
netbsd/386
netbsd/amd64
netbsd/arm
openbsd/386
openbsd/amd64
openbsd/arm
plan9/386
plan9/amd64
plan9/arm
solaris/amd64
windows/386
windows/amd64

本地编译

本地编译就是指编译当前平台可执行程序,我们可以使用以下方式

go build .

依赖 CGO 时

本地化编译时,当依赖 CGO 时,使用上面的方式是会依赖动态链接库的,如果我们想要完全静态化的程序可以使用如下方式:

CGO_ENABLED=0 go build -a -ldflags '-extldflags "-static"' .

交叉编译

交叉编译就是在一种平台上编译出其他平台的可执行程序,比如我们在 Mac 上编译对上面 Go 支持的所有平台上的可执行程序。
Go 对不同平台进行交叉编译

GOOS=windows GOARCH=386 go build . // 编译成windows 32位系统下运行的程序
GOOS=windowns GOARCH=amd64 go build .  // 编译成windows 64位系统下运行的程序
GOOS=linux GOARCH=386 go build . // 编译成linux 32位系统下运行的程序
GOOS=linux GOARCH=amd64 go build . // 编译成linux 64位系统下运行的程序
GOOS=linux  GOARCH=arm GOARM=7 go build . // 编译成在 arm-7上执行的程序

我们只需要用 GOOS指定 OS,GOARCH 指定平台即可(arm需要使用 GOARM 指定特定arm)。

依赖 CGO 时

交叉编译是不支持 CGO 的,当我们依赖库包含对 CGO 的依赖时(比如我们引用了 sqlite3 时),我们就不能使用上面的方法构建完全静态化的程序,因为我们需要编译其他平台的 C 源码,不同平台是不一致的,但是网上可以找到不同的系统的 C 的编译工具来完成交叉编译的完全静态化,但是不同的平台需要不同的工具这样就会变得很复杂。
xgo 就是为了解决交叉编译而生的工具,它提供了各种平台的编译工具来完成完全静态化的编译。
首先安装 xgo 依赖的 docker 环境

> $ docker pull karalabe/xgo-latest

安装 xgo

> $go get github.com/karalabe/xgo

然后就可以使用 xgo 打包了。
参数说明

--targets=linux/arm             : builds only the ARMv5 Linux binaries (arm-6/arm-7 allowed)
--targets=windows/*,darwin/*    : builds all Windows and OSX binaries
--targets=*/arm                 : builds ARM binaries for all platforms
--targets=*/*                   : builds all suppoted targets (default)

打包命令示例

xgo --deps=https://gmplib.org/download/gmp/gmp-6.1.2.tar.bz2 --targets=linux/386 .
xgo --deps=https://gmplib.org/download/gmp/gmp-6.1.2.tar.bz2 --targets=linux/amd64 .
xgo --deps=https://gmplib.org/download/gmp/gmp-6.1.2.tar.bz2 --targets=windows/386 .
xgo --deps=https://gmplib.org/download/gmp/gmp-6.1.2.tar.bz2 --targets=windows/amd64 .
xgo --deps=https://gmplib.org/download/gmp/gmp-6.1.2.tar.bz2 --targets=darwin/amd64 .

跟多内容参考xgo文档。

作者头像

刘宇帅

非著名程序员,全栈开发工程师,长期专注系统开发与架构设计。

提示

功能待开通!


暂无评论~

相关文章

震惊!同事小张踩了 gorm 神奇的 Scan 函数的坑

gorm 简介 gorm 是 go 语言中实现的比较好的 ORM 包,且是国人开发的。项目地址 事故描述 Scan 是 gorm 提供的一个把数据库结果读取到 struct 的函数。定义如下: // Scan scan value to a struct func (s *DB) Scan(dest interface{}) *DB { return s.NewScope(s.Value).Set("gorm:query_destination", dest).callCallbacks(s.parent.callbacks.queries).db } 今天同事小张写代码的时候写了一个

goland集成fmt goimports gometalinter

三个工具介绍 go fmt是用来规范go文件格式,比如格式化单个文件 go fmt xxx.go goimports 用来检查导入包,导入依赖包,删除不依赖的包 gometalinter 集成go语言几乎所有检测工具,静态分析代码,包含功能如下 go vet -工具可以帮我们静态分析我们的源码存在的各种问题,例如多余的代码,提前return的逻辑,struct的tag是否符合标准等。 go tool vet --shadow -用来检查作用域里面设置的局部变量名和全局变量名设置一样导致全局变量设置无效的问题 gotype -类型检测用来检测传递过来的变量和预期变量类型一致 gotype -x

[转]Golang中JSON使用小技巧

临时忽略掉struct中空字段 type User struct { Email string `json:"email"` Password string `json:"password"` } 当我们把用户信息返回给前端的时候显然需要忽略调Password 字段,则可以这样做: json.Marshal(struct{ *User Password bool `json:"password,omitempty"` }{ User:user, }) 临时添加额外字段 type User struct { Email string `json:"

golang 中丰富的字符串格式化

golang中字符串格式化输出 package main import ( "fmt" "os" ) type point struct { x, y int } func main() { // Go提供了几种打印格式,用来格式化一般的Go值,例如 // 下面的%v打印了一个point结构体的对象的值 p := point{1, 2} fmt.Printf("%v\n", p) // 如果所格式化的值是一个结构体对象,那么`%+v`的格式化输出 // 将包括结构体的成员名称和值 fmt.Printf("%

Go 条件编译

Go语言的条件编译是由go/build包支持的,通过条件编译我们可以实现根据不同的参数编译包里不同的文件。 Go是怎么支持条件编译的 Go通过在源代码里添加编译标签(build tag)实现条件编译的。编译标签是以// +build开始,并且出现在代码文件的最开始。构建选项规则如下: 以逗号分割的选项是并的关系 以空格分割的选项是或的关系 条件项的名字用字母+数字表示,!表示否定的意思 构建标签后必须留一行空行 例如如下的例子: // +build linux,386 darwin,!cgo 对应的布尔表达式就是 (linux AND 386) OR (darwin AND (NOT cg