当前位置: 代码网 > it编程>前端脚本>Golang > Go语言中命令行参数解析工具pflag的使用指南

Go语言中命令行参数解析工具pflag的使用指南

2024年11月25日 Golang 我要评论
在使用 go 进行开发的过程中,命令行参数解析是我们经常遇到的需求。尽管 go 标准库提供了 flag 包用于实现命令行参数解析,但只能满足基本需要,不支持高级特性。于是 go 社区中出现了一个叫 p

在使用 go 进行开发的过程中,命令行参数解析是我们经常遇到的需求。尽管 go 标准库提供了 flag 包用于实现命令行参数解析,但只能满足基本需要,不支持高级特性。于是 go 社区中出现了一个叫 pflag 的第三方包,功能更加全面且足够强大。在本文中,我们将学习并掌握如何使用 pflag。

特点

pflag 作为 go 内置 flag 包的替代品,具有如下特点:

  • 实现了 posix/gnu 风格的 --flags。
  • pflag 与《the gnu c library》 中「25.1.1 程序参数语法约定」章节中 posix 建议语法兼容。
  • 兼容 go 标准库中的 flag 包。如果直接使用 flag 包定义的全局 flagset 对象 commandline,则完全兼容;否则当你手动实例化了 flagset 对象,这时就需要为每个标志设置一个简短标志(shorthand)。

使用

基本用法

我们可以像使用 go 标准库中的 flag 包一样使用 pflag。

package main

import (
 "fmt"

 "github.com/spf13/pflag"
)

type host struct {
 value string
}

func (h *host) string() string {
 return h.value
}

func (h *host) set(v string) error {
 h.value = v
 return nil
}

func (h *host) type() string {
 return "host"
}

func main() {
 var ip *int = pflag.int("ip", 1234, "help message for ip")

 var port int
 pflag.intvar(&port, "port", 8080, "help message for port")

 var h host
 pflag.var(&h, "host", "help message for host")

 // 解析命令行参数
 pflag.parse()

 fmt.printf("ip: %d\n", *ip)
 fmt.printf("port: %d\n", port)
 fmt.printf("host: %+v\n", h)

 fmt.printf("nflag: %v\n", pflag.nflag()) // 返回已设置的命令行标志个数
 fmt.printf("narg: %v\n", pflag.narg())   // 返回处理完标志后剩余的参数个数
 fmt.printf("args: %v\n", pflag.args())   // 返回处理完标志后剩余的参数列表
 fmt.printf("arg(1): %v\n", pflag.arg(1)) // 返回处理完标志后剩余的参数列表中第 i 项
}

以上示例演示的 pflag 用法跟 flag 包用法一致,可以做到二者无缝替换。

示例分别使用 pflag.int()pflag.intvar()pflag.var() 三种不同方式来声明标志。其中 ipport 都是 int 类型标志,host 标志则为自定义的 host 类型,它实现了 pflag.value 接口,通过实现接口类型,标志能够支持任意类型,增加灵活性。

通过 --help/-h 参数查看命令行程序使用帮助:

$ go run main.go --help                      
usage of ./main:
      --host host   help message for host
      --ip int      help message for ip (default 1234)
      --port int    help message for port (default 8080)
pflag: help requested

可以发现,帮助信息中的标志位置是经过重新排序的,并不是标志定义的顺序。

与 flag 包不同的是,pflag 包参数定界符是两个 -,而不是一个 -,在 pflag 中 --- 具有不同含义,这点稍后会进行介绍。

ip 标志的默认参数为 1234port 标志的默认参数为 8080

注意:在有些终端下执行程序退出后,还会多打印一行 exit status 2,这并不意味着程序没有正常退出,而是因为 --help 意图就是用来查看使用帮助,所以程序在打印使用帮助信息后,主动调用 os.exit(2) 退出了。

通过如下方式使用命令行程序:

$ go run main.go --ip 1 x y --host localhost a b 
ip: 1
port: 8080
host: {value:localhost}
nflag: 2
narg: 4
args: [x y a b]
arg(1): y

ip 标志的默认值已被命令行参数 1 所覆盖,由于没有传递 port 标志,所以打印结果为默认值 8080host 标志的值也能够被正常打印。

还有 4 个非选项参数数 xyab 也都被 pflag 识别并记录了下来。这点比 flag 要强大,在 flag 包中,非选项参数数只能写在所有命令行参数最后,xy 出现在这里程序是会报错的。

进阶用法

除了像 flag 包一样的用法,pflag 还支持一些独有的用法,以下是用法示例。

package main

import (
 "fmt"
 "os"

 "github.com/spf13/pflag"
)

type host struct {
 value string
}

func (h *host) string() string {
 return h.value
}

func (h *host) set(v string) error {
 h.value = v
 return nil
}

func (h *host) type() string {
 return "host"
}

func main() {
 flagset := pflag.newflagset("test", pflag.exitonerror)

 var ip = flagset.intp("ip", "i", 1234, "help message for ip")

 var boolvar bool
 flagset.boolvarp(&boolvar, "boolvar", "b", true, "help message for boolvar")

 var h host
 flagset.varp(&h, "host", "h", "help message for host")

 flagset.sortflags = false

 flagset.parse(os.args[1:])

 fmt.printf("ip: %d\n", *ip)
 fmt.printf("boolvar: %t\n", boolvar)
 fmt.printf("host: %+v\n", h)

 i, err := flagset.getint("ip")
 fmt.printf("i: %d, err: %v\n", i, err)
}

首先我们通过 pflag.newflagset 自定义了 flagset 对象 flagset,之后的标志定义和解析都通过 flagset 来完成。

前文示例中 pflag.int() 这种用法,实际上使用的是全局 flagset 对象 commandlinecommandline 定义如下:

var commandline = newflagset(os.args[0], exitonerror)

现在同样使用三种不同方式来声明标志,分别为 flagset.intp()flagset.boolvarp()flagset.varp()。不难发现,这三个方法的命名结尾都多了一个 p,它们的能力也得以升级,三个方法都多了一个 shorthand string 参数(flagset.intp 的第 2 个参数,flagset.boolvarpflagset.varp 的第 3 个参数)用来设置简短标志。

从声明标志的方法名中我们能够总结出一些规律:

  • pflag.<type> 类方法名会将标志参数值存储在指针中并返回。
  • pflag.<type>var 类方法名中包含 var 关键字的,会将标志参数值绑定到第一个指针类型的参数。
  • pflag.<type>ppflag.<type>varp 类方法名以 p 结尾的,支持简短标志。

一个完整标志在命令行传参时使用的分界符为 --,而一个简短标志的分界符则为 -

flagset.sortflags = false 作用是禁止打印帮助信息时对标志进行重排序。

示例最后,使用 flagset.getint() 获取参数的值。

通过 --help/-h 参数查看命令行程序使用帮助:

$ go run main.go --help
usage of test:
  -i, --ip int      help message for ip (default 1234)
  -b, --boolvar     help message for boolvar (default true)
  -h, --host host   help message for host
pflag: help requested

这次的帮助信息中,标志顺序没有被改变,就是声明的顺序。

每一个标志都会对应一个简短标志,如 -b--boolvar 是等价的,可以更加方便的设置参数。

指定如下命令行参数运行示例:

$ go run main.go --ip 1 -h localhost --boolvar=false
ip: 1
boolvar: false
host: {value:localhost}
i: 1, err: <nil>

通过 --ip 1 使用完整标志指定 ip 参数值。

通过 -h localhost 使用简短标志指定 host 参数值。

布尔类型的标志指定参数 --boolvar=false 需要使用等号 = 而非空格。

命令行标志语法

命令行标志遵循如下语法:

语法说明
--flag适用于 bool 类型标志,或具有 nooptdefval 属性的标志。
--flag x适用于非 bool 类型标志,或没有 nooptdefval 属性的标志。
--flag=x适用于 bool 类型标志。
-n 1234/-n=1234/-n1234简短标志,非 bool 类型且没有 nooptdefval 属性,三者等价。

标志解析在终止符 -- 之后停止。

整数标志接受 1234、0664、0x1234,并且可能为负数。

布尔标志接受 1, 0, t, f, true, false, true, false, true, false。

duration 标志接受任何对 time.parseduration 有效的输入。

标志名 normalize

借助 pflag.normalizedname 我们能够给标志起一个或多个别名、规范化标志名等。

package main

import (
 "fmt"
 "os"
 "strings"

 "github.com/spf13/pflag"
)

func normalizefunc(f *pflag.flagset, name string) pflag.normalizedname {
 // alias
 switch name {
 case "old-flag-name":
  name = "new-flag-name"
  break
 }

 // --my-flag == --my_flag == --my.flag
 from := []string{"-", "_"}
 to := "."
 for _, sep := range from {
  name = strings.replace(name, sep, to, -1)
 }
 return pflag.normalizedname(name)
}

func main() {
 flagset := pflag.newflagset("test", pflag.exitonerror)

 var ip = flagset.intp("new-flag-name", "i", 1234, "help message for new-flag-name")
 var myflag = flagset.intp("my-flag", "m", 1234, "help message for my-flag")

 flagset.setnormalizefunc(normalizefunc)
 flagset.parse(os.args[1:])

 fmt.printf("ip: %d\n", *ip)
 fmt.printf("myflag: %d\n", *myflag)
}

要使用 pflag.normalizedname,我们需要创建一个函数 normalizefunc,然后将其通过 flagset.setnormalizefunc(normalizefunc) 注入到 flagset 使其生效。

normalizefunc 函数中,我们给 new-flag-name 标志起了一个别名 old-flag-name

另外,还对标志名进行了规范化处理,带有 -_ 分割符的标志名,会统一规范化成以 . 作为分隔符的标志名。

使用示例如下:

$ go run pflag.go --old-flag-name 2 --my-flag 200
ip: 2
myflag: 200

$ go run pflag.go --new-flag-name 3 --my_flag 300
ip: 3
myflag: 300

nooptdefval

nooptdefvalno option default values 的简写。

创建标志后,可以为标志设置 nooptdefval 属性,如果标志具有 nooptdefval 属性并且在命令行上设置了标志而没有参数选项,则标志将设置为 nooptdefval 指定的值。

如下示例:

var ip = flag.intp("flagname", "f", 1234, "help message")
flag.lookup("flagname").nooptdefval = "4321"

不同参数结果如下:

命令行参数结果值
--flagname=1357ip=1357
--flagnameip=4321
[nothing]ip=1234

弃用/隐藏标志

使用 flags.markdeprecated 可以弃用一个标志,使用 flags.markshorthanddeprecated 可以弃用一个简短标志,使用 flags.markhidden 可以隐藏一个标志。

package main

import (
 "fmt"
 "os"

 "github.com/spf13/pflag"
)

func main() {
 flags := pflag.newflagset("test", pflag.exitonerror)

 var ip = flags.intp("ip", "i", 1234, "help message for ip")

 var boolvar bool
 flags.boolvarp(&boolvar, "boolvar", "b", true, "help message for boolvar")

 var h string
 flags.stringvarp(&h, "host", "h", "127.0.0.1", "help message for host")

 // 弃用标志
 flags.markdeprecated("ip", "deprecated")
 flags.markshorthanddeprecated("boolvar", "please use --boolvar only")

 // 隐藏标志
 flags.markhidden("host")

 flags.parse(os.args[1:])

 fmt.printf("ip: %d\n", *ip)
 fmt.printf("boolvar: %t\n", boolvar)
 fmt.printf("host: %+v\n", h)
}

查看使用帮助:

$ go run main.go -h                                 
usage of test:
      --boolvar   help message for boolvar (default true)
pflag: help requested

从打印结果可以发现,弃用标志 ip 时,其对应的简短标志 i 也会跟着被弃用;弃用 boolvar 所对应的简短标志 b 时,boolvar 标志会被保留;host 标志则完全被隐藏。

指定如下命令行参数运行示例:

$ go run main.go --ip 1 --boolvar=false -h localhost
flag --ip has been deprecated, deprecated
ip: 1
boolvar: false
host: localhost

打印信息中会提示用户 ip 标志已经弃用,不过使用 --ip 1 指定的参数值依然能够生效。

隐藏的 host 标志使用 -h localhost 指定参数值同样能够生效。

指定如下命令行参数运行示例:

$ go run main.go -i 1 -b=false --host localhost
flag --ip has been deprecated, deprecated
flag shorthand -b has been deprecated, please use --boolvar only
ip: 1
boolvar: false
host: localhost

打印信息中增加了一条简短标志 -b 已被弃用的提示,指定参数值依然生效。

对于弃用的 ip 标志,使用简短标志形式传惨 -i 1 同样生效。

支持 flag 类型

由于 pflag 对 flag 包兼容,所以可以在一个程序中混用二者:

package main

import (
 "flag"
 "fmt"

 "github.com/spf13/pflag"
)

func main() {
 var ip *int = pflag.int("ip", 1234, "help message for ip")
 var port *int = flag.int("port", 80, "help message for port")

 pflag.commandline.addgoflagset(flag.commandline)
 pflag.parse()

 fmt.printf("ip: %d\n", *ip)
 fmt.printf("port: %d\n", *port)
}

其中,ip 标志是使用 pflag.int() 声明的,port 标志则是使用 flag.int() 声明的。只需要通过 addgoflagset 方法将 flag.commandline 注册到 pflag 中,那么 pflag 就可以使用 flag 中声明的标志集合了。

运行示例结果如下:

$ go run main.go --ip 10 --port 8000
ip: 10
port: 8000

总结

本文主要介绍了 go第三方标志包 pflag 的特点及用法。

首先介绍了 pflag 的基本使用方法,包括声明标志、解析命令行参数、获取标志值等。接着介绍了 pflag 的进阶用法,例如自定义 flagset、使用 pflag.<type>p 方法来支持简短标志。之后又对命令行标志语法进行了讲解,对于布尔值、非布尔值和简短标志,都有各自不同的语法。我们还讲解了如何借助 pflag.normalizedname 给标志起一个或多个别名、规范化标志名。然后介绍了 nooptdefval 的作用和如何弃用/隐藏标志。最后通过示例演示了如何在一个程序中混用 flag 和 pflag。

彩蛋:不知道你有没有发现,示例中的 ip 标志的名称其实代表的是 int pointer 而非 internet protocol addressip 标志源自官方示例,不过我顺势而为又声明了 porthost 标志,算是一个程序中的谐音梗 :)。

以上就是go语言中命令行参数解析工具pflag的使用指南的详细内容,更多关于go pflag命令行参数解析的资料请关注代码网其它相关文章!

(0)

相关文章:

版权声明:本文内容由互联网用户贡献,该文观点仅代表作者本人。本站仅提供信息存储服务,不拥有所有权,不承担相关法律责任。 如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 2386932994@qq.com 举报,一经查实将立刻删除。

发表评论

验证码:
Copyright © 2017-2025  代码网 保留所有权利. 粤ICP备2024248653号
站长QQ:2386932994 | 联系邮箱:2386932994@qq.com