代码生成利器 - Go Generate

简介: 代码生成利器 - Go Generate

简介

  在Go语言中,go generate​ 命令是一个非常有用的工具,它可以帮助我们自动化地生成代码。本文将详细介绍如何使用 go generate​ 命令,并提供一些示例来说明它的用法。

什么是Generate命令?

  ​go generate​ 命令是一个用于自动化生成Go代码的工具。它可以在Go源文件中的特殊注释中指定命令,然后在运行 go generate​ 命令时自动执行这些命令。这些命令可以用来生成代码、格式化代码、运行测试等等。

  在Go 1.4版本中,go generate​ 命令首次发布,它的目的是为了解决Go语言中的一些重复性工作。自从发布以来,它已经成为了一个非常受欢迎的工具,广泛应用于Go语言的开发中。

语法

go generate [-run regexp] [-n] [-v] [-x] [build flags] [file.go... | packages]
  • generate 由现有指令描述的运行命令文件。这些命令可以运行任何进程,但目的是创建或更新 Go 源文件
  • Go generate永远不会由 go build、go test 等等。它必须显式运行

  注释格式:

  在Go源文件中使用 go generate​ 命令需要在源代码中添加特殊注释,格式如下:

  其中, command​ 是要执行的命令, arguments​ 是传递给该命令的参数。

//go:generate command argument...

Generate命令的使用示例

  例如,假设我们有一个名为 hello.go​ 的源文件,其中包含以下注释:

//go:generate echo "Hello, World!"

  当我们在命令行中运行 go generate​ 命令时,它将自动执行上述注释中的命令,并在控制台中输出 "Hello, World!"。

Generate生成顺序

//go:generate echo "top"
package main

import "fmt"

//go:generate echo "middle"
func main() {
    fmt.Println("hello, go generate")
}

//go:generate echo "tail"

$go generate main.go
top
middle
tail

正则匹配生成

  go generate还可以通过-run使用正则式去匹配各源文件中go generate指示符中的命令,并仅执行匹配成功的命令:

// 未匹配到任何go generate指示符中的命令
$go generate -x -v -run "protoc" ./...
main.go
subpkg1/subpkg1.go
subpkg2/subpkg2.go

Generate应用场景

生成枚举类型的String方法

  这里我们需要安装stringer工具

 go install  golang.org/x/tools/cmd/stringer@latest

  假如有如下枚举

type BACnetComfirmedServiceChoice byte

const (
    ServiceConfirmedAcknowledgeAlarm           BACnetComfirmedServiceChoice = 0 //确认报警服务
    ServiceConfirmedComfirmedCOVNotification   BACnetComfirmedServiceChoice = 1 //证实COV通告服务
    ServiceConfirmedComfirmedEventNotification BACnetComfirmedServiceChoice = 2 //证实事件通过服务
}

  通常我们会为枚举类型手写String方法,这样在打印上面枚举常量时能输出有意义的内容:

func (d BACnetComfirmedServiceChoice ) String() string {
    switch d {
    case ServiceConfirmedAcknowledgeAlarm           :
        return "确认报警服务"
    case ServiceConfirmedComfirmedCOVNotification   :
        return "证实COV通告服务"
    case ServiceConfirmedComfirmedEventNotification:
        return "证实事件通过服务"
    }

    return "确认报警服务"
}

  如果一个项目中枚举常量类型有很多,逐个为其手写String方法费时费力。当枚举常量有变化的时候,手动维护String方法十分烦琐且易错。对于这种情况,使用go generate驱动stringer工具为这些枚举类型自动生成String方法的实现不失为一个较为理想的方案。下面就是利用go generate对上面示例的改造:

//go:generate stringer -type=BACnetComfirmedServiceChoice -linecomment
type BACnetComfirmedServiceChoice byte

const (
    ServiceConfirmedAcknowledgeAlarm           BACnetComfirmedServiceChoice = 0 //确认报警服务
    ServiceConfirmedComfirmedCOVNotification   BACnetComfirmedServiceChoice = 1 //证实COV通告服务
    ServiceConfirmedComfirmedEventNotification BACnetComfirmedServiceChoice = 2 //证实事件通过服务
}

  利用generate生成代码如下:

// Code generated by "stringer -type=BACnetComfirmedServiceChoice -linecomment"; DO NOT EDIT.

package generate

import "strconv"

func _() {
    // An "invalid array index" compiler error signifies that the constant values have changed.
    // Re-run the stringer command to generate them again.
    var x [1]struct{}
    _ = x[ServiceConfirmedAcknowledgeAlarm-0]
    _ = x[ServiceConfirmedComfirmedCOVNotification-1]
    _ = x[ServiceConfirmedComfirmedEventNotification-2]
}

const _BACnetComfirmedServiceChoice_name = "确认报警服务证实COV通告服务证实事件通过服务"

var _BACnetComfirmedServiceChoice_index = [...]uint8{0, 18, 39, 63}

func (i BACnetComfirmedServiceChoice) String() string {
    if i >= BACnetComfirmedServiceChoice(len(_BACnetComfirmedServiceChoice_index)-1) {
        return "BACnetComfirmedServiceChoice(" + strconv.FormatInt(int64(i), 10) + ")"
    }
    return _BACnetComfirmedServiceChoice_name[_BACnetComfirmedServiceChoice_index[i]:_BACnetComfirmedServiceChoice_index[i+1]]
}

protobuf

  通过generate将protoc预生成指令写在代码中

package main

import (
    "fmt"
    msg "go-generate/protobuf/msg"
)

//go:generate protoc -I ./IDL msg.proto --gofast_out=./msg
func main() {
    var m = msg.Request{
        MsgID:  "xxxx",
        Field1: "field1",
        Field2: []string{"field2-1", "field2-2"},
    }
    fmt.Println(m)
}

ebpf

  在cilium/ebpf中使用generate调用其cmd下的bpf2go进行编译

/ $BPF_CLANG and $BPF_CFLAGS are set by the Makefile.
//go:generate go run github.com/cilium/ebpf/cmd/bpf2go -cc $BPF_CLANG -cflags $BPF_CFLAGS bpf xdp.c -- -I../headers

func main() {
    ....
}

总结

  ​go generate​ 命令是一个非常有用的工具,它可以帮助我们自动化地生成代码。在本文中,我们介绍了如何在Go源代码中使用 go generate​ 命令,并提供了一些示例来说明它的用法。使用 go generate​ 命令可以提高我们的开发效率,减少我们的重复性工作。

  ‍

参考资料

  1. 命令 go - Go 编程语言 (google.cn)
  2. Generating code - The Go Programming Language
  3. gostringer 命令 - github.com/sourcegraph/gostringer - go 包
  4. ebpf/main.go at master · cilium/ebpf (github.com)
相关文章
|
Go
这个报错是因为在构建chaosblade时,找不到crypto/ecdh模块
【1月更文挑战第22天】【1月更文挑战第110篇】这个报错是因为在构建chaosblade时,找不到crypto/ecdh模块
533 6
|
编译器 Go API
go generate指南:代码自动生成
go generate指南:代码自动生成
3900 0
|
NoSQL 安全 Java
分布式锁实现原理与最佳实践
在单体的应用开发场景中涉及并发同步时,大家往往采用Synchronized(同步)或同一个JVM内Lock机制来解决多线程间的同步问题。而在分布式集群工作的开发场景中,就需要一种更加高级的锁机制来处理跨机器的进程之间的数据同步问题,这种跨机器的锁就是分布式锁。接下来本文将为大家分享分布式锁的最佳实践。
|
Linux Shell 网络安全
Linux 用户管理命令
本文详细介绍了Linux系统中的各类常用命令,包括用户管理(如`adduser`, `usermod`, `passwd`等)、系统操作(如关机、重启、注销)、磁盘管理(如`df`, `mkfs`, `mount`)及网络管理(如`ifconfig`, `ping`, `ssh`)等。通过具体示例展示了每个命令的基本用法和应用场景,帮助用户更好地理解和掌握Linux系统的管理和操作技巧。
|
JavaScript 前端开发 关系型数据库
针对Mysql转义反斜杠的解决方案
针对Mysql转义反斜杠的解决方案
362 2
浅谈 Vue3 的模块拆分与 API 重写
浅谈 Vue3 的模块拆分与 API 重写
|
存储 C++
gRPC 四模式之 双向流RPC模式
gRPC 四模式之 双向流RPC模式
1019 0
|
网络协议 Docker 容器
Docker——如何修改运行中容器的映射端口
Docker——如何修改运行中容器的映射端口
624 0
|
数据安全/隐私保护
vulnhub-GoldenEye靶机通关wp(一)
vulnhub-GoldenEye靶机通关wp
582 0
|
JSON API 数据格式
curl 使用:命令行中的 HTTP 客户端
`curl` 是命令行神器,用于与服务器交互,支持HTTP、HTTPS、FTP等协议。本文教你如何用它做POST请求(-d/--data)、GET请求(-G/-d)、JSON请求(-H 'Content-Type: application/json')和文件上传(-F)。学会这些,轻松测试API、自动化任务。现在就动手试试吧!
440 0