今天遇到了需要使用条件编译的场景,于是查了一下 golang 是如何支持条件编译的。

条件编译简介

go 文档里称之为,Build Constraints,即,编译限制。 也称为 build tag。

//go:build tag_name

其决定了当前文件是否会被当前 package 所包含。

重点:

用于限制一整个文件是否应该被编译入最终的二进制文件,而不是一个文件中的部分代码片段 (block)

build tag 的一些使用场景

  • 测试环境使用 mock 服务;而正式环境使用真实数据
  • 免费版、专业版和企业版提供不同的功能
  • 不同操作系统的兼容性处理。通常用于跨平台,例如 windows,linux,mac 不同兼容处理逻辑。
  • go 低版本的兼容处理

低版本兼容场景示例

例如,我之前就看到 gin 框架下有个 any.go 文件,由于 any 是 go 1.18 引入的别名, 所以其对低版本的兼容处理就是通过 build tag 实现的。

// Copyright 2022 Gin Core Team. All rights reserved.
// Use of this source code is governed by a MIT style
// license that can be found in the LICENSE file.

//go:build !go1.18
// +build !go1.18

package gin

type any = interface{}

使用方法

例如 pro.go 在 main.go 的基础上新增了两个收费版功能:

//go:build pro

package main

func init() {
  features = append(features,
    "Pro 功能 #1",
    "Pro 功能 #2",
  )
}

编译时,加参数

go build -tags pro

go:build 与 +build 的区别

//go:build

是 Go 1.17 中引入的新条件编译指令格式。它旨在替换

// +build

指令。为何要采用新的格式呢?

对比一下新旧格式的区别就知道了:

//go:build linux && amd64 || darwin
// +build linux,amd64 darwin

显而易见的优势:

  • go:build 这种格式,对 coder 来说,更容易理解其逻辑组合
  • 与 //go:embed 和 //go:generate 这些命令相比较,格式上进行了统一

实战案例

参考

  • https://github.com/golang/go/issues/44484
  • https://www.digitalocean.com/community/tutorials/customizing-go-binaries-with-build-tags
  • https://appliedgo.com/blog/what-s-new-in-go-1-17