go 7年前

Go 条件编译

作者头像 刘宇帅
5103 0

Go语言的条件编译是由go/build包支持的,通过条件编译我们可以实现根据不同的参数编译包里不同的文件。

Go是怎么支持条件编译的

Go通过在源代码里添加编译标签(build tag)实现条件编译的。编译标签是以// +build开始,并且出现在代码文件的最开始。构建选项规则如下:

  1. 以逗号分割的选项是并的关系
  2. 以空格分割的选项是或的关系
  3. 条件项的名字用字母+数字表示,!表示否定的意思
  4. 构建标签后必须留一行空行

例如如下的例子:

// +build linux,386 darwin,!cgo

对应的布尔表达式就是

(linux AND 386) OR (darwin AND (NOT cgo))

一个文件可能由多个build构建标签,那么多个构建标签之间是并的关系例如:

// +build linux darwin
// +build 386

对应的布尔表达式是

(linux OR darwin) AND 386

我们以json包的使用为例子

Golang官方提供的json包encoding/json,在大部分情况下是足够用了,但是这个包很大的问题就是性能,它不够快。所以有时候我们需要使用更快的包比如json-iterator和easyjson。但是我们在修改的时候直接替换掉所有的包,所以我们需要做测试,并兼容两个包,所以我们就使用构建标签来做。
我们使用一个自定义的json包来适配encoding/jsonjson-iterator
json/json.go

// +build !jsoniter

package json

import (
    "encoding/json"
    "fmt"
)

func MarshalIndent(v interface{}, prefix, indent string) ([]byte, error) {
    fmt.Println("encoding/json")
    return json.MarshalIndent(v,prefix,indent)
}

json/jsoniter.go

// +build jsoniter

package json

import (
    "fmt"
    "github.com/json-iterator/go"
)

var (
    json = jsoniter.ConfigCompatibleWithStandardLibrary
)

func MarshalIndent(v interface{}, prefix, indent string) ([]byte, error) {
    fmt.Println("jsoniter")
    return json.MarshalIndent(v,prefix,indent)
}

我们main函数如下:

package main

import (
    "fmt"

    "github.com/yushuailiu/easyGolang/buildtag/json"
)

type User struct {
    Name string
}
func main() {
    u := User{Name:"刘宇帅"}
    b,err := json.MarshalIndent(u, "", "")
    if err != nil {
        fmt.Println(err)
    } else {
        fmt.Println(string(b))
    }
}

整个代码结构如下:

json
├── json.go
└── jsoniter.go
main.go

我们直接运行main.go输出如下:

$ go run main.go
encoding/json
{
"Name": "刘宇帅"
}

我们使用标签指定jsoniter:

$go run -tags=jsoniter main.go
jsoniter
{"Name":"刘宇帅"}
作者头像

刘宇帅

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

提示

功能待开通!


暂无评论~

相关文章

震惊!同事小张踩了 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 本地编译和交叉编译

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

[转]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("%