Golang学习笔记--flag包

Golang学习笔记--flag包,第1张

概述原文链接: http://www.cloudnoter.com/?p=131 flag包是用来处理命令参数的。总得来说,其通过将命令行标志与某一具体变量进行绑定,开发人员通过使用该变量进行业务逻辑处理。 一、FlagSet是该包的核心类型: type FlagSet struct { // Usage is the function called when an error occurs

原文链接: http://www.cloudnoter.com/?p=131

flag包是用来处理命令参数的。总得来说,其通过将命令行标志与某一具体变量进行绑定,开发人员通过使用该变量进行业务逻辑处理。

一、FlagSet是该包的核心类型:
type FlagSet struct {    // Usage is the function called when an error occurs while parsing flags.    // The fIEld is a function (not a method) that may be changed to point to    // a custom error handler.    Usage func()    name          string    parsed        bool    actual        map[string]*Flag    // 存放命令行中实际输入的标志映射    formal        map[string]*Flag  // 存放该实例可以处理的命令行标志映射    args          []string // arguments after flags 存放非标志的参数列表(即标志后面的参数)    exitOnError  bool    // does the program exit if there's an error?    errorHandling ErrorHandling    output        io.Writer // nil means stderr; use out() accessor}
该类型同时提供了一系列的方法集合【MethodSet】,通过该方法集用于可以实现灵活的命令行标志处理。


二、flag包export的变量:Commandline
// Commandline is the default set of command-line flags,parsed from os.Args.// The top-level functions such as BoolVar,Arg,and on are wrappers for the// methods of Commandline.var Commandline = NewFlagSet(os.Args[0],ExitOnError)
该包提供了一个默认变量:Commandline,其为FlatSet的一个变量(用面向对象的术语叫做:FlagSet的一个实例)

该flag包export的所有函数本质上都是调用FlagSet类型变量(实例):Commandline的方法实现。如下:
// Int defines an int flag with specifIEd name,default value,and usage string.// The return value is the address of an int variable that stores the value of the flag.func Int(name string,value int,usage string) *int {    return Commandline.Int(name,value,usage)}


三、flag包支持的标志格式
Command line flag @R_419_5135@:        -flag // 代表bool值,相当于-flag=true        -flag=x        -flag x  // non-boolean flags only  不支持bool值标志    One or two minus signs may be used; they are equivalent.  // -flag=value与--flag=value是等效的    The last form is not permitted for boolean flags because the    meaning of the command        cmd -x *    will change if there is a file called 0,false,etc.  You must    use the -flag=false form to turn off a boolean flag.    Flag parsing stops just before the first non-flag argument    ("-" is a non-flag argument) or after the terminator "--".  // 碰到连续两个"-"号且参数长度为2时则终止标志解析


四、标志绑定相关方法:以绑定int类型为例。
// Intvar defines an int flag with specifIEd name,and usage string.// The argument p points to an int variable in which to store the value of the flag.// FlagSet提供的绑定int类型标志的方法,无返回值。通过传入int类型指针变量进行绑定,当调用该方法后,会将绑定信息存入FlagSet.formal映射中func (f *FlagSet) Intvar(p *int,name string,usage string) {    f.Var(newIntValue(value,p),name,usage)}//与上述方法相对应的flag包export的函数:// Intvar defines an int flag with specifIEd name,and usage string.// The argument p points to an int variable in which to store the value of the flag.func Intvar(p *int,usage string) {    Commandline.Var(newIntValue(value,usage)}//使用示例var flagvar intflag.Intvar(&flagvar,"flagname2",1234,"help message for flagname2")// Int defines an int flag with specifIEd name,and usage string.// The return value is the address of an int variable that stores the value of the flag.// FlagSet提供的绑定int类型标志的方法,有返回值,返回int类型指针,当调用该方法后,会将绑定信息存入FlagSet.formal映射中func (f *FlagSet) Int(name string,usage string) *int {    p := new(int)    f.Intvar(p,usage)    return p}//与上述方法相对应的flag包export的函数:// Int defines an int flag with specifIEd name,usage)}//使用示例var flagvar = flag.Int("flagname","help message for flagname")


五、解析标志的相关关键源码
// Parse parses the command-line flags from os.Args[1:].  Must be called// after all flags are defined and before flags are accessed by the program.// flag包export的函数,调用时机为:在设置好标志与变量的绑定关系后,调用flag.Parse()。func Parse() {    // Ignore errors; Commandline is set for ExitOnError.    Commandline.Parse(os.Args[1:])}// Parse parses flag deFinitions from the argument List,which should not// include the command name.  Must be called after all flags in the FlagSet// are defined and before flags are accessed by the program.// The return value will be ErrHelp if -help was set but not defined.// FlagSet类型提供的实现方法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}// parSEOne parses one flag. It reports whether a flag was seen.// 解析每个标志并返回相关结果,若碰到 '-' 或 '--' 时也会直接终止整个标志解析过程,每解析成功一个标志就会将该标志信息放入FlagSet.actual映射中func (f *FlagSet) parSEOne() (bool,error) {    if len(f.args) == 0 {        return false,nil    }    s := f.args[0]    if len(s) == 0 || s[0] != '-' || len(s) == 1 {        return false,nil    }    num_minuses := 1    if s[1] == '-' {        num_minuses++        if len(s) == 2 { // "--" terminates the flags            f.args = f.args[1:]            return false,nil        }    }    name := s[num_minuses:]    if len(name) == 0 || name[0] == '-' || name[0] == '=' {        return false,f.failf("bad flag @R_419_5135@: %s",s)    }    // it's a flag. does it have an argument?    f.args = f.args[1:]    has_value := false    value := ""    for i := 1; i < len(name); i++ { // equals cannot be first        if name[i] == '=' {            value = name[i+1:]            has_value = true            name = name[0:i]            break        }    }    m := f.formal    flag,alreadythere := m[name] // BUG    if !alreadythere {        if name == "help" || name == "h" { // special case for nice help message.            f.usage()            return false,ErrHelp        }        return false,f.failf("flag provIDed but not defined: -%s",name)    }    if fv,ok := flag.Value.(boolFlag); ok && fv.IsBoolFlag() { // special case: doesn't need an arg        if has_value {            if err := fv.Set(value); err != nil {                return false,f.failf("invalID boolean value %q for  -%s: %v",err)            }        } else {            fv.Set("true")        }    } else {        // It must have a value,which might be the next argument.        if !has_value && len(f.args) > 0 {            // value is the next arg            has_value = true            value,f.args = f.args[0],f.args[1:]        }        if !has_value {            return false,f.failf("flag needs an argument: -%s",name)        }        if err := flag.Value.Set(value); err != nil {            return false,f.failf("invalID value %q for flag -%s: %v",err)        }    }    if f.actual == nil {        f.actual = make(map[string]*Flag)    }    f.actual[name] = flag    return true,nil}
总结

以上是内存溢出为你收集整理的Golang学习笔记--flag包全部内容,希望文章能够帮你解决Golang学习笔记--flag包所遇到的程序开发问题。

如果觉得内存溢出网站内容还不错,欢迎将内存溢出网站推荐给程序员好友。

欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/langs/1288881.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2022-06-09
下一篇 2022-06-09

发表评论

登录后才能评论

评论列表(0条)

保存