Go笔记-flag参数解析

先查看godoc文档

first demo

习自 https://github.com/on99/gocyclo/blob/master/gocyclo.go

package flag_demo

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

const usageDoc = `Calculate cyclomatic complexities of Go functions.
Usage:
		gocyclo [flags] <Go file or directory> ...
		
Flags:
		-over N balabala..
		-top N balabala...
		-avg balabala...

...
`

func usage() {
	fmt.Fprintf(os.Stderr, usageDoc)
	os.Exit(2)
}

var (
	over = flag.Int("over", 0, "show functions with complexity > N only")
	top  = flag.Int("top", -1, "show the top N most complex functions only")
	avg  = flag.Bool("avg", false, "show the average complexity")
)

func MyMain() {
	flag.Usage = usage
	flag.Parse()

	args := flag.Args()
	if len(args) == 0 {
		usage()
	}
	fmt.Println("args: ", args)

	fmt.Println("over: ", *over)
	fmt.Println("top: ", *top)
	fmt.Println("avg: ", *avg)
}

使用1:

./src

输出:

Calculate cyclomatic complexities of Go functions.
Usage:
		gocyclo [flags] <Go file or directory> ...
		
Flags:
		-over N balabala..
		-top N balabala...
		-avg balabala...

...

使用2:

./src -over=5 -top=9 -avg=true some thing

输出:

args:  [some thing]
over:  5
top:  9
avg:  true

利用VisitAll自动生成flag信息

package main

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

func main() {
	usage := GenerateUsageFunc()
	ParseFlag(usage)
	fmt.Println(myvar)
}

var myvar string

func init() {
	flag.StringVar(&myvar, "myarg", "def", "whatever...")
}

func GenerateUsageFunc() func() {
	name := os.Args[0] // 程序路径
	return func() {
		fmt.Fprintf(os.Stderr, "Usage of %s:\n", name)
		fmt.Fprintf(os.Stderr, "\t%s [flags] \n", name)
		fmt.Fprintf(os.Stderr, "Flags:\n")
		// 这里就是自动生成 flag 的信息
		flag.VisitAll(func(fg *flag.Flag) {
			fmt.Fprintf(os.Stderr, "  -%s = %s (%s)\n", fg.Name, fg.DefValue, fg.Usage)
		})
	}
}

func ParseFlag(usageFunc func()) {
	// CommandLine 是 flag 包中某些全局函数的默认实现,其实就是一个 FlagSet
	flag.CommandLine.Init(os.Args[0], flag.ExitOnError)
	flag.CommandLine.SetOutput(os.Stderr)
	flag.CommandLine.Usage = usageFunc
	flag.Parse()
	flag.VisitAll(func(fg *flag.Flag) {
		fmt.Printf("your flag: [%10s]=%10s \n", fg.Name, fg.Value.String())
	})
}

输出帮助信息 test.exe -h:

Usage of D:\go\code\test\bin\test.exe:
        D:\go\code\test\bin\test.exe [flags]
Flags:
  -myarg = def (whatever...)

设置正确的参数 test.exe -myarg=a

your flag: [     myarg]=         a
a

设置错误的参数 test.exe -myerr=a

flag provided but not defined: -myerr
Usage of D:\go\code\test\bin\test.exe:
        D:\go\code\test\bin\test.exe [flags]
Flags:
  -myarg = def (whatever...)

命令行参数

http://golang-china.github.io/gopl-zh/ch1/ch1-02.html

os.Args

os.Args这个变量是一个字符串(string)的slice

os.Args的第一个元素,即os.Args[0]是命令行执行时的命令本身;其它的元素则是执行该命令时传给这个程序的参数

os.Args vs flag.Args

func main() {
	flag.Parse()
	fmt.Println(os.Args)
	fmt.Println(flag.NArg(), flag.Args())
}

执行命令:

./test cmd1 cmd2 -arg1=v1 -arg2=v2

执行结果:

[./test cmd1 cmd2 -arg1=v1 -arg2=v2]
4 [cmd1 cmd2 -arg1=v1 -arg2=v2]

os.Argsflag.Args 多了一个运行路径,但 flag 需要先 parse

如果添加了下面的代码:

func init() {
	flag.String("arg1", "", "")
	flag.String("arg2", "", "")
}

再次执行 ./test cmd1 cmd2 -arg1=v1 -arg2=v2 时输出的就是:

[./test -arg1=v1 -arg2=v2 cmd1 cmd2]
2 [cmd1 cmd2]

flag.Args() 没有返回已经被解析的 -arg1-arg2

flag set

var (
	flagset = flag.NewFlagSet("test", flag.ExitOnError)
	arg1    = flagset.String("arg1", "arg1value", "usage of arg1")
)

func main() {
	flagset.Parse(os.Args[1:])
	fmt.Println(*arg1)
}

lookup获取的值的类型

var flagset = flag.NewFlagSet("hello", flag.ExitOnError)

func init() {
	flagset.Int("a", 1, "usage")
}

func main() {
	flagset.Parse(os.Args[1:])
	vv := flagset.Lookup("a").Value.(flag.Getter).Get().(int)
	fmt.Println(vv)
}

多个flag set

package main

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

var flagset = flag.NewFlagSet("test", flag.ExitOnError)
var flagset2 = flag.NewFlagSet("test2", flag.ExitOnError)

func init() {
	flagset.String("arg1", "value1", "haha")
	flagset2.String("arg1", "value2", "hehe")
}

func main() {
	flagset.Parse(os.Args[1:])
	flagset2.Parse(os.Args[1:])

	f := flagset.Lookup("arg1")
	fmt.Println(f.Value)

	f = flagset2.Lookup("arg1")
	fmt.Println(f.Value)
}

usage 只显示第一个flagset的:

Usage of test:
  -arg1 string
        haha (default "value1")

并且设置了arg1后两个flagset都是可以获取到新值的。

-arg1=abc

如果两个flagset的变量一个是arg1,一个是arg2,那么设置了其中一个,另一个就会报错,没有定义这个变量,所以,flagset只能用一个

转载于:https://my.oschina.net/u/2004526/blog/847195

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
你的问题是关于Gson下划线转驼峰注解的使用吗?下划线和驼峰式命名是两种常见的命名规则,在Java开发中也经常会使用到。Gson是Google开发的一个Java库,用于将Java对象转换为JSON格式的数据。在使用Gson进行JSON解析时,如果JSON中的属性名采用了下划线命名,则可以使用Gson提供的@SerializedName注解进行转换。 使用@SerializedName注解需要指定一个value值,用于表示JSON中的属性名,同时还可以指定一个alternate值,用于表示其他可能的属性名。例如,如果JSON中的属性名为"user_name",则可以使用@SerializedName("user_name")注解将其转换为Java对象中的"userName"属性。 另外,如果Java对象中的属性名采用了驼峰式命名,则可以使用Gson提供的FieldNamingStrategy接口进行转换。通过实现FieldNamingStrategy接口中的方法,可以自定义属性名的转换规则。例如,可以实现一个驼峰式转下划线的命名策略: ``` public class CamelCaseToUnderscoresNamingStrategy implements FieldNamingStrategy { @Override public String translateName(Field field) { String name = field.getName(); StringBuilder result = new StringBuilder(); for (int i = 0; i < name.length(); i++) { char ch = name.charAt(i); if (Character.isUpperCase(ch)) { result.append("_").append(Character.toLowerCase(ch)); } else { result.append(ch); } } return result.toString(); } } ``` 使用时可以通过GsonBuilder设置: ``` Gson gson = new GsonBuilder() .setFieldNamingStrategy(new CamelCaseToUnderscoresNamingStrategy()) .create(); ``` 这样,在解析JSON时就可以将驼峰式命名的属性名转换为下划线命名了。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值