缘起

我一直在想,有什么方式可以让人比较轻易地保持每日学习,持续输出的状态。写博客是一种方式,但不是每天都有想写的,值得写的东西。

有时候一个技术比较复杂,写博客的时候经常会写着写着发现自己的理解有偏差,或者细节还没有完全掌握,要去查资料,了解了之后又继续写,如此反复。

这样会导致一篇博客的耗时过长。

我在每天浏览思否掘金Github的过程中,发现一些比较好的想法,有JS 每日一题NodeJS 每日一库每天一道面试题等等等等。

https://github.com/parro-it/awesome-micro-npm-packages这个仓库收集 NodeJS 小型库,一天看一个不是梦!这也是我这个系列的灵感。

我计划每天学习一个 Go 语言的库,输出一篇介绍型的博文。每天一库当然是理想状态,我心中的预期是一周 3-5 个。

今天是第一天,我们从一个基础库聊起————Go 标准库中的flag

简介

flag用于解析命令行选项。有过类 Unix 系统使用经验的童鞋对命令行选项应该不陌生。例如命令ls -al列出当前目录下所有文件和目录的详细信息,其中-al就是命令行选项。

命令行选项在实际开发中很常用,特别是在写工具的时候。

  • 指定配置文件的路径,如redis-server ./redis.conf以当前目录下的配置文件redis.conf启动 Redis 服务器;
  • 自定义某些参数,如python -m SimpleHTTPServer 8080启动一个 HTTP 服务器,监听 8080 端口。如果不指定,则默认监听 8000 端口。

快速使用

学习一个库的第一步当然是使用它。我们先看看flag库的基本使用:

package main

import (
"fmt"
"flag"
) var (
intflag int
boolflag bool
stringflag string
) func init() {
flag.IntVar(&intflag, "intflag", 0, "int flag value")
flag.BoolVar(&boolflag, "boolflag", false, "bool flag value")
flag.StringVar(&stringflag, "stringflag", "default", "string flag value")
} func main() {
flag.Parse() fmt.Println("int flag:", intflag)
fmt.Println("bool flag:", boolflag)
fmt.Println("string flag:", stringflag)
}

可以先编译程序,然后运行(我使用的是 Win10 + Git Bash):

$ go build -o main.exe main.go
$ ./main.exe -intflag 12 -boolflag 1 -stringflag test

输出:

int flag: 12
bool flag: true
string flag: test

如果不设置某个选项,相应变量会取默认值:

$ ./main.exe -intflag 12 -boolflag 1

输出:

int flag: 12
bool flag: true
string flag: default

可以看到没有设置的选项stringflag为默认值default

还可以直接使用go run,这个命令会先编译程序生成可执行文件,然后执行该文件,将命令行中的其它选项传给这个程序。

$ go run main.go -intflag 12 -boolflag 1

可以使用-h显示选项帮助信息:

$ ./main.exe -h
Usage of D:\code\golang\src\github.com\darjun\cmd\flag\main.exe:
-boolflag
bool flag value
-intflag int
int flag value
-stringflag string
string flag value (default "default")

总结一下,使用flag库的一般步骤:

  • 定义一些全局变量存储选项的值,如这里的intflag/boolflag/stringflag
  • init方法中使用flag.TypeVar方法定义选项,这里的Type可以为基本类型Int/Uint/Float64/Bool,还可以是时间间隔time.Duration。定义时传入变量的地址、选项名、默认值和帮助信息;
  • main方法中调用flag.Parseos.Args[1:]中解析选项。因为os.Args[0]为可执行程序路径,会被剔除。

注意点:

flag.Parse方法必须在所有选项都定义之后调用,且flag.Parse调用之后不能再定义选项。如果按照前面的步骤,基本不会出现问题。

因为init在所有代码之前执行,将选项定义都放在init中,main函数中执行flag.Parse时所有选项都已经定义了。

选项格式

flag库支持三种命令行选项格式。

-flag
-flag=x
-flag x

---都可以使用,它们的作用是一样的。有些库使用-表示短选项,--表示长选项。相对而言,flag使用起来更简单。

第一种形式只支持布尔类型的选项,出现即为true,不出现为默认值。

第三种形式不支持布尔类型的选项。因为这种形式的布尔选项在类 Unix 系统中可能会出现意想不到的行为。看下面的命令:

cmd -x *

其中,*是 shell 通配符。如果有名字为 0、false的文件,布尔选项-x将会取false。反之,布尔选项-x将会取true。而且这个选项消耗了一个参数。

如果要显示设置一个布尔选项为false,只能使用-flag=false这种形式。

遇到第一个非选项参数(即不是以---开头的)或终止符--,解析停止。运行下面程序:

$ ./main.exe noflag -intflag 12

将会输出:

int flag: 0
bool flag: false
string flag: default

因为解析遇到noflag就停止了,后面的选项-intflag没有被解析到。所以所有选项都取的默认值。

运行下面的程序:

$ ./main.exe -intflag 12 -- -boolflag=true

将会输出:

int flag: 12
bool flag: false
string flag: default

首先解析了选项intflag,设置其值为 12。遇到--后解析终止了,后面的--boolflag=true没有被解析到,所以boolflag选项取默认值false

解析终止之后如果还有命令行参数,flag库会存储下来,通过flag.Args方法返回这些参数的切片。

可以通过flag.NArg方法获取未解析的参数数量,flag.Arg(i)访问位置i(从 0 开始)上的参数。

选项个数也可以通过调用flag.NFlag方法获取。

稍稍修改一下上面的程序:

func main() {
flag.Parse() fmt.Println(flag.Args())
fmt.Println("Non-Flag Argument Count:", flag.NArg())
for i := 0; i < flag.NArg(); i++ {
fmt.Printf("Argument %d: %s\n", i, flag.Arg(i))
} fmt.Println("Flag Count:", flag.NFlag())
}

编译运行该程序:

$ go build -o main.exe main.go
$ ./main.exe -intflag 12 -- -stringflag test

输出:

[-stringflag test]
Non-Flag Argument Count: 2
Argument 0: -stringflag
Argument 1: test

解析遇到--终止后,剩余参数-stringflag test保存在flag中,可以通过Args/NArg/Arg等方法访问。

整数选项值可以接受 1234(十进制)、0664(八进制)和 0x1234(十六进制)的形式,并且可以是负数。实际上flag在内部使用strconv.ParseInt方法将字符串解析成int

所以理论上,ParseInt接受的格式都可以。

布尔类型的选项值可以为:

  • 取值为true的:1、t、T、true、TRUE、True;
  • 取值为false的:0、f、F、false、FALSE、False。

另一种定义选项的方式

上面我们介绍了使用flag.TypeVar定义选项,这种方式需要我们先定义变量,然后变量的地址。

还有一种方式,调用flag.Type(其中Type可以为Int/Uint/Bool/Float64/String/Duration等)会自动为我们分配变量,返回该变量的地址。用法与前一种方式类似:

package main

import (
"fmt"
"flag"
) var (
intflag *int
boolflag *bool
stringflag *string
) func init() {
intflag = flag.Int("intflag", 0, "int flag value")
boolflag = flag.Bool("boolflag", false, "bool flag value")
stringflag = flag.String("stringflag", "default", "string flag value")
} func main() {
flag.Parse() fmt.Println("int flag:", *intflag)
fmt.Println("bool flag:", *boolflag)
fmt.Println("string flag:", *stringflag)
}

编译并运行程序:

$ go build -o main.exe main.go
$ ./main.exe -intflag 12

将输出:

int flag: 12
bool flag: false
string flag: default

除了使用时需要解引用,其它与前一种方式基本相同。

高级用法

定义短选项

flag库并没有显示支持短选项,但是可以通过给某个相同的变量设置不同的选项来实现。即两个选项共享同一个变量。

由于初始化顺序不确定,必须保证它们拥有相同的默认值。否则不传该选项时,行为是不确定的。

package main

import (
"fmt"
"flag"
) var logLevel string func init() {
const (
defaultLogLevel = "DEBUG"
usage = "set log level value"
) flag.StringVar(&logLevel, "log_type", defaultLogLevel, usage)
flag.StringVar(&logLevel, "l", defaultLogLevel, usage + "(shorthand)")
} func main() {
flag.Parse() fmt.Println("log level:", logLevel)
}

编译、运行程序:

$ go build -o main.exe main.go
$ ./main.exe -log_type WARNING
$ ./main.exe -l WARNING

使用长、短选项均输出:

log level: WARNING

不传入该选项,输出默认值:

$ ./main.exe
log level: DEBUG

解析时间间隔

除了能使用基本类型作为选项,flag库还支持time.Duration类型,即时间间隔。时间间隔支持的格式非常之多,例如"300ms"、"-1.5h"、"2h45m"等等等等。

时间单位可以是 ns/us/ms/s/m/h/day 等。实际上flag内部会调用time.ParseDuration。具体支持的格式可以参见time(需fq)库的文档。

package main

import (
"flag"
"fmt"
"time"
) var (
period time.Duration
) func init() {
flag.DurationVar(&period, "period", 1*time.Second, "sleep period")
} func main() {
flag.Parse()
fmt.Printf("Sleeping for %v...", period)
time.Sleep(period)
fmt.Println()
}

根据传入的命令行选项period,程序睡眠相应的时间,默认 1 秒。编译、运行程序:

$ go build -o main.exe main.go
$ ./main.exe
Sleeping for 1s... $ ./main.exe -period 1m30s
Sleeping for 1m30s...

自定义选项

除了使用flag库提供的选项类型,我们还可以自定义选项类型。我们分析一下标准库中提供的案例:

package main

import (
"errors"
"flag"
"fmt"
"strings"
"time"
) type interval []time.Duration func (i *interval) String() string {
return fmt.Sprint(*i)
} func (i *interval) Set(value string) error {
if len(*i) > 0 {
return errors.New("interval flag already set")
}
for _, dt := range strings.Split(value, ",") {
duration, err := time.ParseDuration(dt)
if err != nil {
return err
}
*i = append(*i, duration)
}
return nil
} var (
intervalFlag interval
) func init() {
flag.Var(&intervalFlag, "deltaT", "comma-seperated list of intervals to use between events")
} func main() {
flag.Parse() fmt.Println(intervalFlag)
}

首先定义一个新类型,这里定义类型interval

新类型必须实现flag.Value接口:

// src/flag/flag.go
type Value interface {
String() string
Set(string) error
}

其中String方法格式化该类型的值,flag.Parse方法在执行时遇到自定义类型的选项会将选项值作为参数调用该类型变量的Set方法。

这里将以,分隔的时间间隔解析出来存入一个切片中。

自定义类型选项的定义必须使用flag.Var方法。

编译、执行程序:

$ go build -o main.exe main.go
$ ./main.exe -deltaT 30s
[30s]
$ ./main.exe -deltaT 30s,1m,1m30s
[30s 1m0s 1m30s]

如果指定的选项值非法,Set方法返回一个error类型的值,Parse执行终止,打印错误和使用帮助。

$ ./main.exe -deltaT 30x
invalid value "30x" for flag -deltaT: time: unknown unit x in duration 30x
Usage of D:\code\golang\src\github.com\darjun\go-daily-lib\flag\self-defined\main.exe:
-deltaT value
comma-seperated list of intervals to use between events

解析程序中的字符串

有时候选项并不是通过命令行传递的。例如,从配置表中读取或程序生成的。这时候可以使用flag.FlagSet结构的相关方法来解析这些选项。

实际上,我们前面调用的flag库的方法,都会间接调用FlagSet结构的方法。flag库中定义了一个FlagSet类型的全局变量CommandLine专门用于解析命令行选项。

前面调用的flag库的方法只是为了提供便利,它们内部都是调用的CommandLine的相应方法:

// src/flag/flag.go
var CommandLine = NewFlagSet(os.Args[0], ExitOnError) func Parse() {
CommandLine.Parse(os.Args[1:])
} func IntVar(p *int, name string, value int, usage string) {
CommandLine.Var(newIntValue(value, p), name, usage)
} func Int(name string, value int, usage string) *int {
return CommandLine.Int(name, value, usage)
} func NFlag() int { return len(CommandLine.actual) } func Arg(i int) string {
return CommandLine.Arg(i)
} func NArg() int { return len(CommandLine.args) }

同样的,我们也可以自己创建FlagSet类型变量来解析选项。

package main

import (
"flag"
"fmt"
) func main() {
args := []string{"-intflag", "12", "-stringflag", "test"} var intflag int
var boolflag bool
var stringflag string fs := flag.NewFlagSet("MyFlagSet", flag.ContinueOnError)
fs.IntVar(&intflag, "intflag", 0, "int flag value")
fs.BoolVar(&boolflag, "boolflag", false, "bool flag value")
fs.StringVar(&stringflag, "stringflag", "default", "string flag value") fs.Parse(args) fmt.Println("int flag:", intflag)
fmt.Println("bool flag:", boolflag)
fmt.Println("string flag:", stringflag)
}

NewFlagSet方法有两个参数,第一个参数是程序名称,输出帮助或出错时会显示该信息。第二个参数是解析出错时如何处理,有几个选项:

  • ContinueOnError:发生错误后继续解析,CommandLine就是使用这个选项;
  • ExitOnError:出错时调用os.Exit(2)退出程序;
  • PanicOnError:出错时产生 panic。

随便看一眼flag库中的相关代码:

// src/flag/flag.go
func (f *FlagSet) Parse(arguments []string) error {
f.parsed = true
f.args = arguments
for {
seen, err := f.parseOne()
if seen {
continue
}
if err == nil {
break
}
switch f.errorHandling {
case ContinueOnError:
return err
case ExitOnError:
os.Exit(2)
case PanicOnError:
panic(err)
}
}
return nil
}

与直接使用flag库的方法有一点不同,FlagSet调用Parse方法时需要显示传入字符串切片作为参数。因为flag.Parse在内部调用了CommandLine.Parse(os.Args[1:])

示例代码都放在GitHub上了。

参考

  1. flag库文档

我的博客

欢迎关注我的微信公众号【GoUpUp】,共同学习,一起进步~

本文由博客一文多发平台 OpenWrite 发布!

Go 每日一库之 flag的更多相关文章

  1. Go 每日一库之 go-flags

    简介 在上一篇文章中,我们介绍了flag库.flag库是用于解析命令行选项的.但是flag有几个缺点: 不显示支持短选项.当然上一篇文章中也提到过可以通过将两个选项共享同一个变量迂回实现,但写起来比较 ...

  2. Go 每日一库之 viper

    简介 上一篇文章介绍 cobra 的时候提到了 viper,今天我们就来介绍一下这个库. viper 是一个配置解决方案,拥有丰富的特性: 支持 JSON/TOML/YAML/HCL/envfile/ ...

  3. Go 每日一库之 fsnotify

    简介 上一篇文章Go 每日一库之 viper中,我们介绍了 viper 可以监听文件修改进而自动重新加载. 其内部使用的就是fsnotify这个库,它是跨平台的.今天我们就来介绍一下它. 快速使用 先 ...

  4. Go内置库模块 flag

    import "flag" flag包实现了命令行参数的解析.每个参数认为一条记录,根据实际进行定义,到一个set集合.每条都有各自的状态参数. 在使用flag时正常流程: 1.  ...

  5. Go基础篇【第6篇】: 内置库模块 flag

    import "flag" flag包实现了命令行参数的解析.每个参数认为一条记录,根据实际进行定义,到一个set集合.每条都有各自的状态参数. 在使用flag时正常流程: 1.  ...

  6. Go语言标准库之flag

    Go语言内置的flag包实现了命令行参数的解析,flag包使得开发命令行工具更为简单. os.Args 如果你只是简单的想要获取命令行参数,可以像下面的代码示例一样使用os.Args来获取命令行参数. ...

  7. Go 每日一库之 go-homedir

    简介 今天我们来看一个很小,很实用的库go-homedir.顾名思义,go-homedir用来获取用户的主目录. 实际上,使用标准库os/user我们也可以得到这个信息: package main i ...

  8. Go 每日一库之 go-ini

    简介 ini 是 Windows 上常用的配置文件格式.MySQL 的 Windows 版就是使用 ini 格式存储配置的. go-ini是 Go 语言中用于操作 ini 文件的第三方库. 本文介绍g ...

  9. Go 每日一库之 cobra

    简介 cobra是一个命令行程序库,可以用来编写命令行程序.同时,它也提供了一个脚手架, 用于生成基于 cobra 的应用程序框架.非常多知名的开源项目使用了 cobra 库构建命令行,如Kubern ...

随机推荐

  1. win10 uwp 使用 Azure DevOps 自动构建

    通过 Azure DevOps 可以做到自动构建程序,覆盖计划.创建.编程.测试.部署.发布.托管.共享等各个环节,适用于大多数的语言.平台. 本文继续使用图床为例告诉大家如何使用 Azure Dev ...

  2. P1086 最大素数积

    题目描述 我们称一个整数 \(x\) 是"素数积"当且仅当 \(x = a \times b\) 并且 \(a\) 和 \(b\) 都是素数. 现在告诉你一个数 \(N(1 \le ...

  3. RocketMQ(消息重发、重复消费、事务、消息模式)

    分布式开放消息系统(RocketMQ)的原理与实践 RocketMQ基础:https://github.com/apache/rocketmq/tree/rocketmq-all-4.5.1/docs ...

  4. Java动态编译优化——提升编译速度(N倍)

    一.前言 最近一直在研究Java8 的动态编译, 并且也被ZipFileIndex$Entry 内存泄漏所困扰,在无意中,看到一个第三方插件的动态编译.并且编译速度是原来的2-3倍.原本打算直接用这个 ...

  5. 【39.68%】【CF 714 C】Filya and Homework

    time limit per test 1 second memory limit per test 256 megabytes input standard input output standar ...

  6. 【codeforces 764A】Taymyr is calling you

    time limit per test1 second memory limit per test256 megabytes inputstandard input outputstandard ou ...

  7. CF 453C. Little Pony and Summer Sun Celebration

    CF 453C. Little Pony and Summer Sun Celebration 构造题. 题目大意,给定一个无向图,每个点必须被指定的奇数或者偶数次,求一条满足条件的路径(长度不超\( ...

  8. 中文javadoc

    大量的中文注释,其实最大的目的是想生成中文的Javadoc.但是中文注释实在太不高级了,在各种编码下还会变成乱码.本着高端大气国际化的思路,还是想把它换成英文,但是又不想放弃中文的Javadoc,怎么 ...

  9. k8s的持久化存储

    本例使用nfs 创建pv [root@k8s-master data]# vi pv.yaml apiVersion: v1kind: PersistentVolumemetadata: name: ...

  10. Servlet 的面试题

    Servlet运行在Servlet容器中,其生命周期由容器来管理.Servlet的生命周期通过javax.servlet.Servlet接口中的init().service()和destroy()方法 ...