- ubuntu12.04环境下使用kvm ioctl接口实现最简单的虚拟机
- Ubuntu 通过无线网络安装Ubuntu Server启动系统后连接无线网络的方法
- 在Ubuntu上搭建网桥的方法
- ubuntu 虚拟机上网方式及相关配置详解
CFSDN坚持开源创造价值,我们致力于搭建一个资源共享平台,让每一个IT人在这里找到属于你的精彩世界.
这篇CFSDN的博客文章Golang开发命令行之flag包的使用方法由作者收集整理,如果你对这篇文章有兴趣,记得点赞哟.
日常命令行操作,相对应的众多命令行工具是提高生产力的必备工具,鼠标能够让用户更容易上手,降低用户学习成本。 而对于开发者,键盘操作模式能显著提升生产力,还有在一些专业工具中, 大量使用快捷键代替繁琐的鼠标操作,能够使开发人员更加专注于工作,提高效率,因为键盘操作模式更容易产生肌肉记忆 。
举个栗子:我司业务研发,前些年在我们的强力推动下(被迫)转向使用了 git 作为版本控制,开始使用的是图形化“小乌龟”工具。后续出现几次问题解决起来较麻烦后,推荐其使用原生的 git 命令行。如今,使用 git 命令行操作版本控制可谓 “一顿操作猛如虎......” 。
命令行(键盘)操作在很大程度上可以提高工作效率,与之相对应的是鼠标(触屏等)操作,这两种模式是目前的主流人机交互方式 。
设计一款命令行工具的开发语言可以选择原始的 shell 、甚至是更原始的语言 C ,更为容易上手且功能更多的有 node 、 python 、 golang 。
本文是基于 golang 开发命令行工具的开篇,主要是基于 golang 原生内置的、轻量的 flag 包实现,用 golang 设计命令行工具而不用 shell 、 python 的原因这里就不做论述了 。
flag 包用来解析命令行参数 。
相比简单的使用 os.Args 来获取命令行参数, flag 可以实现按照更为通用的命令行用法,例如 mysql -u root -p 123456 。其中 mysql 是命令行的名称即这个命令, -u 和 -p 分别是这个命令的两个参数:用户名和密码,后面接着的是对应的参数值,有了参数的声明之后,两个参数可以互换位置,参数值也可以选填或按照缺省(默认)值进行指定 。
flag 包支持的命令行参数的类型有 bool 、 int 、 int64 、 uint 、 uint64 、 float float64 、 string 、 duration 。
即布尔值、整型、浮点型、字符串、时间段类型 。
定义 flag 命令行参数,用来接收命令行输入的参数值,一般有以下两种方法 。
flag.TypeVar():先定义参数(实际上是指针),再定义 flag.TypeVar 将命令行参数存储(绑定)到前面参数的值的指针(地址) 。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
|
var name string
var age int
var height float64
var graduated bool
//
&name 就是接收用户命令行中输入的-n后面的参数值
//
返回值是一个用来存储name参数的值的指针/地址
//
定义string类型命令行参数name,括号中依次是变量名、flag参数名、默认值、参数说明
flag.StringVar(&name,
"n"
,
""
,
"name参数,默认为空"
)
//
定义整型命令行参数age
flag.IntVar(&age,
"a"
, 0,
"age参数,默认为0"
)
//
定义浮点型命令行参数height
flag.Float64Var(&height,
"h"
, 0,
"height参数,默认为0"
)
//
定义布尔型命令行参数graduated
flag.BoolVar(&graduated,
"g"
,
false
,
"graduated参数,默认为false"
)
|
flag.Type():用短变量声明的方式定义参数类型及变量名 。
1
2
3
4
5
6
7
8
|
//
定义string类型命令行参数name,括号中依次是flag参数名、默认值、参数说明
namePtr := flag.String(
"n"
,
""
,
"name参数,默认为空"
)
//
定义整型命令行参数age
age := flag.Int(
"a"
, 0,
"age参数,默认为0"
)
//
定义浮点型命令行参数height
height := flag.Float64(
"h"
, 0,
"height参数,默认为0"
)
//
定义布尔型命令行参数graduated
graduated:= flag.Bool(
"g"
,
false
,
"graduated参数,默认为false"
)
|
固定用法,定义好参数后,通过调用 flag.Parse() 来对命令行参数进行解析写入注册的 flag 里,进而解析获取参数值,通过查看源码中也是调用的 os.Args 。
源码路径 go/src/flag/flag.go 。
1
2
3
4
5
6
|
//
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.
func Parse() {
//
Ignore errors; CommandLine is
set
for
ExitOnError.
CommandLine.Parse(os.Args[1:])
}
|
进而查看 Parse 方法的源码 。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
|
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:
if
err == ErrHelp {
os.Exit(0)
}
os.Exit(2)
case
PanicOnError:
panic(err)
}
}
return
nil
}
|
真正解析参数的是 parseOne 方法(这里省略源码),结论是 。
解析参数时,对于参数的指定方式一般有"-"、"--"、以及是否空格等方式,组合下来有如下几种方式 。
。
-flag xxx | 空格和一个 - 符号 |
---|---|
--flag xxx | 空格和两个 - 符号 |
-flag=xxx | 等号和一个 - 符号 |
--flag=xxx | 等号和两个 - 符号 |
。
其中, -flag xxx 方式最为常用,如果参数是布尔型,只能用等号方式指定 。
flag 包默认会根据定义的命令行参数,在使用时如果不输入参数就打印对应的帮助信息 。
这样的帮助信息我们可以对其进行覆盖去改变默认的 Usage 。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
|
package main
import
(
"flag"
"fmt"
)
func main() {
var host string
var port int
var verbor bool
var help bool
//
绑定命令行参数与变量关系
flag.StringVar(&host,
"H"
,
"127.0.0.1"
,
"ssh host"
)
flag.IntVar(&port,
"P"
, 22,
"ssh port"
)
flag.BoolVar(&verbor,
"v"
,
false
,
"detail log"
)
flag.BoolVar(&help,
"h"
,
false
,
"help"
)
//
自定义-h
flag.Usage = func() {
fmt
.Println(`
Usage: flag [-H addr] [-p port] [-
v
]
Options:
`)
flag.PrintDefaults()
}
//
解析命令行参数
flag.Parse()
if
help {
flag.Usage()
}
else
{
fmt
.Println(host, port, verbor)
}
}
/*
➜ go run flag_args.go -h
Usage: flag [-H addr] [-p port] [-
v
]
Options:
-H string
ssh
host (default
"127.0.0.1"
)
-P int
ssh
port (default 22)
-h help
-
v
detail log
*/
|
简单来说,短参数和长参数,就是例如我们在使用某些命令时,查看命令版本可以输入 -V ,也可以输入 --version 。这种情况下, flag 并没有默认支持,但是可以通过可以两个选项共享同一个变量来实现,即通过给某个相同的变量设置不同的选项,参数在初始化的时候其顺序是不固定的,因此还需要保证其拥有相同的默认值 。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
|
package main
import
(
"fmt"
"flag"
)
var logLevel string
func init() {
const (
defaultLogLevel =
"DEBUG"
usage =
"set log level"
)
flag.StringVar(&logLevel,
"log_level"
, defaultLogLevel, usage)
flag.StringVar(&logLevel,
"l"
, defaultLogLevel, usage +
"(shorthand)"
)
}
func main() {
flag.Parse()
fmt
.Println(
"log level:"
, logLevel)
}
|
通过 const 声明公共的常量,并在默认值以及帮助信息中去使用,这样就可以实现了 。
实现计算字符串或目录下递归计算文件 md5 的命令,类似 linux 的 md5sum 命令 。
其中利用 bufio 分批次读取文件,防止文件过大时造成资源占用高 。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
|
package main
import
(
"bufio"
"crypto/md5"
"flag"
"fmt"
"io"
"os"
"strings"
)
func md5reader(reader *bufio.Reader) string {
//
hasher := md5.New()
//
定义MD5
hash
计算器
bytes :=
make
([]byte, 1024*1024*10)
//
分批次读取文件
for
{
n, err := reader.Read(bytes)
if
err != nil {
if
err != io.EOF {
return
""
}
break
}
else
{
hasher.Write(bytes[:n])
}
}
return
fmt
.Sprintf(
"%x"
, hasher.Sum(nil))
}
func md5file(path string) (string, error) {
file
, err := os.Open(path)
if
err != nil {
return
""
, err
}
else
{
defer
file
.Close()
return
md5reader(bufio.NewReader(
file
)), nil
}
}
func md5str(txt string) (string, error) {
return
md5reader(bufio.NewReader(strings.NewReader(txt))), nil
//return
fmt
.Sprintf(
"%x"
, md5.Sum([]byte(txt)))
}
func main() {
txt := flag.String(
"s"
,
""
,
"md5 txt"
)
path := flag.String(
"f"
,
""
,
"file path"
)
help := flag.Bool(
"h"
,
false
,
"help"
)
flag.Usage = func() {
fmt
.Println(`
Usage: md5 [-s 123abc] [-f path]
Options:
`)
flag.PrintDefaults()
}
flag.Parse()
if
*help || *txt ==
""
&& *path ==
""
{
flag.Usage()
}
else
{
var md5 string
var err error
if
*path !=
""
{
md5, err = md5file(*path)
}
else
{
md5, err = md5str(*txt)
}
if
err != nil {
fmt
.Println(err)
}
else
{
fmt
.Println(md5)
}
}
}
|
编译生成二进制文件 。
1
2
3
|
➜ go build -o md5go -x md5_bufio.go
➜ ll md5go
-rwxr-xr-x 1 ssgeek staff 1.9M Oct 2 00:54 md5go
|
测试使用 。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
|
➜ .
/md5go
-h
Usage: md5 [-s 123abc] [-f path]
Options:
-f string
file
path
-h help
-s string
md5 txt
➜ .
/md5go
-s 123456
e10adc3949ba59abbe56e057f20f883e
➜ .
/md5go
-f md5_bufio.go
8607a07cbb98cec0e9abe14b0db0bee6
|
到此这篇关于Golang开发命令行之flag包的使用方法的文章就介绍到这了,更多相关Golang开发命令行之flag包的使用内容请搜索我以前的文章或继续浏览下面的相关文章希望大家以后多多支持我! 。
原文链接:https://www.tuicool.com/articles/Yj67Brr 。
最后此篇关于Golang开发命令行之flag包的使用方法的文章就讲到这里了,如果你想了解更多关于Golang开发命令行之flag包的使用方法的内容请搜索CFSDN的文章或继续浏览相关文章,希望大家以后支持我的博客! 。
我需要一些说明。我可以直接写入 /dev/port 以直接访问并行端口并且它工作正常(我可以打开插入端口连接器的 LED)。但是,我想我可以用 /dev/mem 做同样的事情? (http://tld
按照目前的情况,这个问题不适合我们的问答形式。我们希望答案得到事实、引用或专业知识的支持,但这个问题可能会引发辩论、争论、投票或扩展讨论。如果您觉得这个问题可以改进并可能重新打开,visit the
我使用 Visual C++ 和 Win32 API 学习了 Windows 编程。如今,似乎大多数应用程序都是使用 C# 在 .NET 中开发的。我知道大多数时候 native 代码和托管代码之间没
请耐心等待。我正在制作一个 java 控制台,类似于此处找到的 DragonConsole https://code.google.com/p/dragonconsole/ 。一切都按计划进行,但我想
关闭。这个问题需要更多 focused .它目前不接受答案。 想要改进这个问题吗? 更新问题,使其只关注一个问题 editing this post . 关闭5年前。 Improve this que
Django 的开发服务器表现得很奇怪。访问它的浏览器在加载时卡住,任何退出它的尝试都不起作用。当我点击 control c看似相当,但实际上仍在运行。让它退出的唯一方法是重新启动我的电脑,这很令人沮
我正在使用 Flash Develop,并且创建了一个 ActionScript 3.0 项目。它启动并读取一个 xml 文件,其中包含图像的 url。我已将 url 保留在与 swf 相同的文件夹中
是否可以根据其 website 上提供的规范开发 AUTOSAR BSW 堆栈(例如用于 CAN 通信)?不购买任何昂贵的供应商工具?可以遵循哪些步骤?我被要求探索这种可能性。 最佳答案 是和否。工具
有人知道如何用音频文件的内容覆盖 iPhone 麦克风吗? 想象一个场景,您正在通话,并且想要播放一些简短的音频让其他人听到。 因此,有必要将麦克风(硬件)置于保持状态,并使用委托(delegate)
我遇到了这个问题,我的应用程序出现 EXC_BAD_ACCESS 错误并卡住/停止。我使用模拟器的“向左旋转”和“向右旋转”选项来模拟方向变化行为。导致此错误的可能原因有哪些?由于我没有获得有关错误的
我有超过 1 台 Mac,我想在所有这些 Mac 上进行开发。我知道我需要在每台机器上同步我的手机,但这是我遇到的最小的问题。看起来我无法在手机上运行应用程序,除了在其中之一上开发的应用程序。 是否有
在手机上测试时,我的应用程序在特定点崩溃。控制台显示此消息 Tue Jan 27 15:47:14 unknown SpringBoard[22] : Application com.myprof.
我有一个案例,我从服务器获取信息。我的应用程序有一个选项卡栏和导航按钮。我希望应用程序显示进度指示器并禁用所有其他控件,以便用户在从服务器提取数据时无法跳转。我怎样才能实现这个目标? 我想到的一种方法
有时,当我尝试“构建”/编译下载的源代码时,我会收到以下警告: ld: warning: directory '/Volumes/Skiiing2/CD/ViewBased/Unknown Path/
我无法在 Apple 文档中找到关于开发和分发配置之间差异的明确解释。我目前正在使用开发配置在我的 iPhone 上进行开发和测试。我打算将该应用程序分发到我的 Beta 测试中,我想知道: 我需要使
我在使用 SharePoint 时遇到的最大挑战之一是它不能很好地适应典型的项目环境,其中至少包含开发和生产环境。我遇到的最多的问题是内容和列表是如此紧密地耦合在一起,以至于如果不在生产环境中执行内容
我失败了fist step让 Eclipse(对我来说是全新的)为 ARM 开发做好准备。 我在 Windows 10 中安装了 Eclipse。我想我应该安装 xpm,但我不知道在哪里输入此命令:
首先,我告诉你-我是编码新手 我正在使用vs代码来学习c++,它不会产生像dev c++或codeblocks这样的调试器。我看了一些视频,其中我们必须编辑json文件,这对于初学者来说非常复杂。有人
我失败了fist step让 Eclipse(对我来说是全新的)为 ARM 开发做好准备。 我在 Windows 10 中安装了 Eclipse。我想我应该安装 xpm,但我不知道在哪里输入此命令:
我开发了一个 Ionic 应用程序(iOS 和 Android 的混合)。我有 Xcode 8.3.3 并购买了一年的 Apple Developer Program 订阅。 我不想测试我的应用并将其
我是一名优秀的程序员,十分优秀!