flag 用法参考:

https://golang.google.cn/pkg/flag/


1. 命令源码文件怎样接收参数

package main

import (
    "flag" #导入flag包
    "fmt"
)

var name string

func init(){
    flag.StringVar(&name,"name","everyone","The greting object") # 
}

func main(){
    flag.Parse() #函数flag.Parse用于真正解析命令参数,并把它们的值赋给相应的变量。
    fmt.Printf("hello,%s!\n",name)
}

函数flag.StringVar接受 4 个参数。

第 1 个参数是用于存储该命令参数值的地址,具体到这里就是在前面声明的变量name的地址了,由表达式&name表示。

第 2 个参数是为了指定该命令参数的名称,这里是name。

第 3 个参数是为了指定在未追加该命令参数时的默认值,这里是everyone。

第 4 个函数参数,即是该命令参数的简短说明了,这在打印命令说明时会用到。

go run demo2.go -name="huaihe"
Hello, huaihe!


2. 怎样在运行命令源码文件的时候传入参数,又怎样查看参数的使用说明


如果想查看该命令源码文件的参数说明,可以这样做:

go run demo2.go --help
Usage of /var/folders/4w/lv_wjx0d3b3ff30w0bn_65xc0000gn/T/go-build966482306/b001/exe/demo2: #构建上述命令源码文件时临时生成的可执行文件的完整路径
  -name string
        The greeting object. (default "everyone")
exit status 2


如果我们先构建这个命令源码文件再运行生成的可执行文件,像这样:

go build demo2.go
ls demo2
demo2
./demo2 --help
Usage of ./demo2:
  -name string
        The greeting object. (default "everyone")
./demo2 -name="huaihe"
Hello, huaihe!


3. 怎样自定义命令源码文件的参数使用说明

这有很多种方式,最简单的一种方式就是对变量flag.Usage重新赋值。flag.Usage的类型是func(),即一种无参数声明且无结果声明的函数类型。flag.Usage变量在声明时就已经被赋值了,所以我们才能够在运行命令go run demo2.go --help时看到正确的结果。注意,对flag.Usage的赋值必须在调用flag.Parse函数之前。

package main

import (
    "flag"
    "fmt"
    "os"
)

var name string

func init() {
    flag.StringVar(&name, "name", "everyone", "The greting object")
}

func main() {
    flag.Usage = func() {
        fmt.Fprintf(os.Stderr, "Usage of %s:\n", "question")
        flag.PrintDefaults()
    }
    flag.Parse()
    fmt.Printf("hello,%s!\n", name)
}
go run demo2.go --help
#这里并没有打印临时文件位置,而是打印了os.Stderr,因为此时os.Stderr为空
Usage of question:
  -name string
        The greting object (default "everyone")
exit status 2

再深入一层,我们在调用flag包中的一些函数(比如StringVar、Parse等等)的时候,实际上是在调用flag.CommandLine变量的对应方法。

flag.CommandLine相当于默认情况下的命令参数容器。所以,通过对flag.CommandLine重新赋值,我们可以更深层次地定制当前命令源码文件的参数使用说明。

现在我们把main函数体中的那条对flag.Usage变量的赋值语句注销掉,然后在init函数体的开始处添加如下代码:

package main

import (
    "flag"
    "fmt"
    "os"
)

var name string

func init() {
    flag.CommandLine = flag.NewFlagSet("", flag.ExitOnError)
    flag.CommandLine.Usage = func() {
        fmt.Fprintf(os.Stderr, "Usage of %s:\n", "question")
        flag.PrintDefaults()
    }
    flag.StringVar(&name, "name", "everyone", "The greting object")
}

func main() {
    flag.Parse()
    fmt.Printf("hello,%s!\n", name)
}
go run demo2.go --help
Usage of question:
  -name string
        The greting object (default "everyone")
exit status 2

其输出会与上一次的输出的一致。不过后面这种定制的方法更加灵活。比如,当我们把为flag.CommandLine赋值的那条语句改为:

package main

import (
    "flag"
    "fmt"
    "os"
)

var name string

func init() {
    flag.CommandLine = flag.NewFlagSet("", flag.PanicOnError) //注意这里修改为flag.PanicOnError
    flag.CommandLine.Usage = func() {
        fmt.Fprintf(os.Stderr, "Usage of %s:\n", "question")
        flag.PrintDefaults()
    }
    flag.StringVar(&name, "name", "everyone", "The greting object")
}

func main() {
    flag.Parse()
    fmt.Printf("hello,%s!\n", name)
}
go run demo2.go --help
Usage of question:
  -name string
        The greting object (default "everyone")
panic: flag: help requested

goroutine 1 [running]:
flag.(*FlagSet).Parse(0xc000096120, 0xc000098010, 0x1, 0x1, 0xc000082f88, 0x100534f)
        /usr/local/Cellar/go/1.12.7/libexec/src/flag/flag.go:983 +0xf8
flag.Parse(...)
        /usr/local/Cellar/go/1.12.7/libexec/src/flag/flag.go:998
main.main()
        /Users/daixuan/pinduoduo/go/src/github.com/Golang_Puzzlers/src/puzzlers/article2/q1/demo2.go:21 +0x78
exit status 2

这是由于我们在这里传给flag.NewFlagSet函数的第二个参数值是flag.PanicOnError。flag.PanicOnError和flag.ExitOnError都是预定义在flag包中的常量。

flag.ExitOnError的含义是,告诉命令参数容器,当命令后跟--help或者参数设置的不正确的时候,在打印命令参数使用说明后以状态码2结束当前程序。

状态码2代表用户错误地使用了命令,而flag.PanicOnError与之的区别是在最后抛出“运行时恐慌(panic)”。

上述两种情况都会在我们调用flag.Parse函数时被触发。顺便提一句,“运行时恐慌”是 Go 程序错误处理方面的概念。关于它的抛出和恢复方法,我在本专栏的后续部分中会讲到。


下面再进一步,我们索性不用全局的flag.CommandLine变量,转而自己创建一个私有的命令参数容器。我们在函数外再添加一个变量声明:

我们把对flag.StringVar的调用替换为对cmdLine.StringVar调用,再把flag.Parse()替换为cmdLine.Parse(os.Args[1:])。其中的os.Args[1:]指的就是我们给定的那些命令参数。这样做就完全脱离了flag.CommandLine。*flag.FlagSet类型的变量cmdLine拥有很多有意思的方法

package main

import (
    "flag"
    "fmt"
    "os"
)

var name string

// 方式3。
var cmdLine = flag.NewFlagSet("question", flag.ExitOnError)

func init() {
    // 方式3。
    cmdLine.StringVar(&name, "name", "everyone", "The greeting object.")
}

func main() {
    // 方式3。
    cmdLine.Parse(os.Args[1:])
    fmt.Printf("Hello, %s!\n", name)
}
go run demo3.go --help
Usage of question:
  -name string
        The greeting object. (default "everyone")
exit status 2

go run demo3.go --name="test"
Hello, test!


这样做的好处依然是更灵活地定制命令参数容器。但更重要的是,你的定制完全不会影响到那个全局变量flag.CommandLine。