Go原生在flag包提供了一个命令行工具类,它可以让我们执行类似命令行的赋参操作,经常被运用于工具类,特别是数据处理过程,可以方便我们进行参数可视化注解。

命令行工具

原生flag包

Go原生在flag包提供了一个命令行工具类,它可以让我们执行类似命令行的赋参操作,经常被运用于工具类,特别是数据处理过程,可以方便我们进行参数可视化注解。

flag包提供了多个常用类型的赋值方法,如String, Int, Bool, Float64, Duration等。

  • 通过flag.XXXType()函数可以对参数名称,默认值,描述进行定义
  • flag.parse()用于指定程序开始解析传入的命令行变量
  • 参数-h可以调用flag内置的usage()方法, 把参数描述打印出来

下面列举一个简单的例子。

用法示例:

func main() {

	//声明接收变量, 注意返回的是个指针类型
	wordParam := flag.String("word", "defaultValue", "Desc param 1 for something.")
	numParam := flag.Int("number", 0, "Desc param 2 for integer number.")

	//an existing var declared elsewhere in the program
	ok := AllRight()

	flag.BoolVar(&ok, "ok", false, "Desc value for boolean.")

	flag.Parse()

	log.Printf("Flag for wordParam: %s", *wordParam)
	log.Printf("Flag for wordParam: %d", *numParam)
	log.Printf("Flag for inner bool value : %v", ok)
}

func AllRight() bool {
	return true
}

我们通过command line 传参给可执行程序,输出:

$ ./tool.exe -word=hello -number=6
2020/03/18 10:59:49 Flag for wordParam: hello
2020/03/18 10:59:49 Flag for wordParam: 6
2020/03/18 10:59:49 Flag for inner bool value : false

如果使用-h/--help命令可以调出参数提示,输出:

$ ./tool.exe -h
Usage of E:\Code\GoWork\src\HelloGo\basic\flag\tool.exe:
  -number int
        Desc param 2 for integer number.
  -ok
        Desc value for boolean.
  -word string
        Desc param 1 for something. (default "defaultValue")

flag分级用法

在实际应用环境,调用目标可能有多个,有时我们需要多个命令,多个参数联合起来,用于调用不同的方法,类似于参数调用子命令的参数,
如:./cmd foo -a="a" -b=1或者./cmd bar -c="c" -d=2

Subcommands

原生flag包提供了一个子命令构造方式,NewFlagSet 用于返回子命令的flag,示例如下:

我们定义一个解析子命令方法:

func SubCommands() {
	//param1: 命令参数, param2: 错误退出码
	fooCmd := flag.NewFlagSet("foo", flag.ExitOnError)
	//声明子命令fc, fc2
	fc := fooCmd.String("fc", "default of fc", "foo sub value1")
	fc2 := fooCmd.String("fc2", "default of fc2", "foo sub value2")

	barCmd := flag.NewFlagSet("bar", flag.ExitOnError)
	//声明子命令bc
	bc := barCmd.String("bc", "default of bc", "bar sub b")

	if len(os.Args) < 2 {
		fmt.Println("expected 'foo' or 'bar' subcommands")
		fooCmd.Usage()
		barCmd.Usage()
		os.Exit(1)
	}

	//参数逻辑调用, 使用分支语句
	switch os.Args[1] {
	case "foo":
		fooCmd.Parse(os.Args[2:])
		log.Printf("fc of foo: %s", *fc)
		log.Printf("fc2 of foo: %s", *fc2)
	case "bar":
		barCmd.Parse(os.Args[2:])
		log.Printf("bc of bar: %s", *bc)
	default:
		fmt.Println("expected 'foo' or 'bar' subcommands")
		os.Exit(1)
	}

}

执行命令行提示:

$ ./tool.exe
expected 'foo' or 'bar' subcommands
Usage of foo:
  -fc string
        foo sub value1 (default "default of fc")
  -fc2 string
        foo sub value2 (default "default of fc2")
Usage of bar:
  -bc string
        bar sub b (default "default of bc")

命令调用示例:

$ ./tool.exe foo -fc="fc value" -fc2="fc22222"
2020/03/18 12:54:16 fc of foo: fc value
2020/03/18 12:54:16 fc2 of foo: fc22222

问题延申:

上述方式借助Go的内置flag实现了命令传参,但是有个问题,如上面提到,“在实际应用环境,调用目标可能有多个,有时我们需要多个命令”,对每个调用链都起一个switch分支去写可能十分吃力,我们自己能不能对调用链执行封装,尝试实现类似于flag的功能呢?


自定义分支调用

最近在预览前辈的代码块中,发现一个巧妙的实现方式,在Go中,函数是可以作为参数传入的,我们可以利用这一特性建立一个方法链数组,利用全局函数变量根据入参进行调用,类似于面向对象语言里面的多态,本质上是相同类型的不同表现。

程序实例:

  • 先定义一个cmd结构体,封装了指令/执行方法/提示
/*
	自定义方法调用
 */
//封装命令结构, 指令/执行方法/提示
type cmd struct {
	Name    string
	Process func(args ...string)
	Usage   func() string
}

//全局指令数组
var all []*cmd
  • 根据cmd的规范建立多个实现类,如下面的例子,有wind, sun两个实现方式

wind实现:

//可变参数作为可选入参
func Process(args ...string)  {
	if len(args) < 2 {
		logrus.Error(Usage())
		return
	}

	//Do something
	logrus.Infof("Wind fly from %s, level %s.", args[0], args[1])
}

//参数提示
func Usage() string {
	return "Hint: wind <N/S/W/E> <level>"
}

sun实现:

func Process(args ...string)  {
	if len(args) < 2 {
		logrus.Error(Usage())
		return
	}

	//Do something
	logrus.Infof("Sun %s about %s miles.", args[0], args[1])

}

//参数提示
func Usage() string {
	return "Hint: sun <rise/down> <range>"
}

至此, 不同指令的表现就实现完了,有点类似于面向对象语言里面的多态,本质上是相同类型的不同表现。
后面我们来看下在main方法怎么调用

func init()  {
	all = append(all, &cmd{"sun", sun.Process, sun.Usage})
	all = append(all, &cmd{"wind", wind.Process, wind.Usage})
}

//全局用法
func usage() string {
	sb := new(strings.Builder)
	sb.WriteString(fmt.Sprintf("Usage: %s <command> [args...]\n", os.Args[0]))
	for _, c := range all {
		sb.WriteString("\n命令: ")
		sb.WriteString(c.Name)
		sb.WriteString(", 用法: ")
		sb.WriteString(c.Usage())
		sb.WriteString("\n")
	}
	return sb.String()
}

func main()  {
	if len(os.Args) > 1 && os.Args[1] != "--help" && os.Args[1] != "-h"{
		for _, c := range all {
			//匹配全局命令
			if c.Name == os.Args[1] {
				c.Process(os.Args[2:]...)
				return
			}
		}
		fmt.Fprintf(os.Stderr, "No match func for %s.\n", os.Args[1])
	}
	fmt.Fprintln(os.Stderr, usage())
}

我们利用入参的长度以及内置命令区分调用链,对符合格式的参数与全局指令切片进行匹配。
下面是 输出示例:

$ ./tool.exe -h
Usage: E:\Code\GoWork\src\HelloGo\basic\flag\tool.exe <command> [args...]

命令: sun, 用法: Hint: sun <rise/down> <range>

命令: wind, 用法: Hint: wind <N/S/W/E> <level>

$ ./tool.exe a b
No match func for a.
Usage: E:\Code\GoWork\src\HelloGo\basic\flag\tool.exe <command> [args...]

命令: sun, 用法: Hint: sun <rise/down> <range>

命令: wind, 用法: Hint: wind <N/S/W/E> <level>

可以看到当参数不合法时会遍历打印各个子命令的用法给予提示。

$ ./tool.exe sun
time="2020-03-18T15:23:23+08:00" level=error msg="Hint: sun <rise/down> <range>"

常规调用:

$ ./tool.exe wind N 7
time="2020-03-18T15:22:40+08:00" level=info msg="Wind fly from N, level 7."
$ ./tool.exe sun rise 30
time="2020-03-18T15:24:28+08:00" level=info msg="Sun rise about 30 miles."

小结

以上是go命令行工具的简单用法,内置的flag帮我们封装好了一下基础函数,我们也可以利用字符串处理自定义实现工具类,兼容多种场景,简单的逻辑判断也能达到flag的那种用法。

参考链接:

Git project
https://github.com/pixeldin/HelloGo/tree/master/basic/flag
Go by Example: Command-Line Subcommands
https://gobyexample.com/command-line-subcommands
Command line flag syntax
https://golang.org/pkg/flag/#hdr-Command_line_flag_syntax