Go语言 命令行解析(一)
命令行啟動服務的方式,在后端使用非常廣泛,如果有寫過C語言的同學相信不難理解這一點!在C語言中,我們可以根據argc和argv來獲取和解析命令行的參數,從而通過不同的參數調取不同的方法,同時也可以用Usage來打印幫助信息了。
那么開始今天的話題之前,我們回顧一下在C語言中是如何解析傳遞的參數的。
示例代碼:
#include <stdio.h> #include <stdlib.h>// argc 為int型 // argv 為char指針數組,元素個數是argc,存放的是指向每一個參數的指針 int main(int argc, char *argv[]) {printf("命令行參數個數: %d\n", argc);printf("執行程序的名稱:%s\n", argv[0]);int i = 1; // 從下標1開始獲取,因為0代表的是程序名稱while(i < argc){// 循環打印每個命令行的參數printf("%s\n", argv[i]);i++;}return 0; }編譯運行:
#gcc c_cli.c -o c_cli ./c_cli /usr/local/service/config /usr/local/service/log/service.log 命令行參數個數: 3 執行程序的名稱:./c_cli /usr/local/service/config /usr/local/service/log/service.log看完上面的例子,其實我們可以發現,上例中是C語言自帶的參數解析,對于簡單的參數構成還是可以使用的。下面我們再看一下Go語言os標準庫的實現。
示例代碼:
package mainimport ("fmt""os" )func main() {var args = os.Argsfmt.Println(args)return }編譯執行:
#go build go_flag.go #./go_flag /usr/local/service/config /usr/local/service/log/service.log [./go_flag /usr/local/service/config /usr/local/service/log/service.log]上例中,我們可以看到os.Args返回一個數組,數組里面是我們命令行執行時,所傳遞的參數和程序名稱。os自帶的參數獲取,對于簡單的參數來說還能使用,如果參數復雜的情況下,那么解析起來就比較費勁的。這個時候,我們可以選擇Go語言的flag標準庫來幫我們處理命令行解析工作。
Flag包:https://golang.org/pkg/flag/
是Go語言提供的一個標準庫,能夠較為方便和靈活的解析命令行傳遞的參數。
flag有兩種方式:
1、flag.Type,其中Type可以是:int、string、bool,float等類型,返回指針類型。
var port = flag.Int("port", 0, "相關描述")參數1:flag的名稱
參數2:flag的值,上例中默認值是0
參數3:flag的描述
2、flag.TypeVar,將類型綁定到一個變量上。
var port int flag.IntVar(&port, "port", 0, "相關描述")參數1:flag的值
參數2:flag的名稱
參數3:flag的值,上例中默認值是0
參數4:flag的描述
示例代碼:
package mainimport ("flag""fmt""os" )var (// 定義一個tcp端口號tcp_port int// flag.Type(Name為Flag的名字,Value是Flag的值,Usage是Flag的提示信息)port = flag.Int("port", 0, "服務端口設置參數為:-port=80")config = flag.String("config", "", "配置文件配置參數為:-config=/usr/local/service/config")logFile = flag.String("logfile", "", "日志文件配置參數為:-logfile=/usr/local/service/log/service.log")// flag.Type返回的是指針類型,所以獲取值為"*變量")func init() {flag.IntVar(&tcp_port, "tcp_port", 0, "TCP服務端口描述: -tcp_port=2001")flag.Usage = func() {_, _ = fmt.Fprint(os.Stderr,"cli : go_flag -port=8080 -config=/usr/local/service/config "+"-logfile=/usr/local/service/log/service.log -tcp_port=2001\n")flag.PrintDefaults()}flag.Parse() }func main() {if *port <= 0 {flag.PrintDefaults()os.Exit(1)}if tcp_port <= 0 {flag.PrintDefaults()os.Exit(1)}if *config == "" {flag.PrintDefaults()os.Exit(2)}if *logFile == "" {flag.PrintDefaults()os.Exit(3)}fmt.Printf("service tcp port : %d \n", tcp_port)fmt.Printf("service port : %d \n", *port)fmt.Printf("service config : %s \n", *config)fmt.Printf("service logfile : %s \n", *logFile)return }運行結果:
#go build go_flag.go #./go_flag -config string配置文件配置參數為:-config=/usr/local/service/config-logfile string日志文件配置參數為:-logfile=/usr/local/service/log/service.log-port int服務端口設置參數為:-port=80-tcp_port intTCP服務端口描述: -tcp_port=2001#./go_flag -port 8090 -config=/usr/local/service/config -logfile= /usr/local/service/log/service.log -tcp_port=2001 service tcp port : 2001 service port : 8090 service config : /usr/local/service/config service logfile : /usr/local/service/log/service.log查看幫助:
#./go_flag -h cli : go_flag -port=8080 -config=/usr/local/service/config -logfile=/usr/local/service/log/service.log-config string配置文件配置參數為:-config=/usr/local/service/config-logfile string日志文件配置參數為:-logfile=/usr/local/service/log/service.log-port int服務端口設置參數為:-port=80-tcp_port intTCP服務端口描述: -tcp_port=2001Flag語法:
- -flag?只支持bool參數
- -flag=p?
- -flag?p?bool類型不能使用,當p為false時會引起歧義
Flag解析:
parseOne()函數來處理-flag=value的,如果處理成功后,會將flag存儲到FlagSet.actucal map[string]*Flag中。
flag.Parse()函數來解析命令行中的參數中定義的flag,該方法遇到第一個非flag的命令方法就中止。
1、non-flag的時候,會終止解析工作。
2、連續兩個“-”的時候,會終止解析工作。
然后,再看我們傳遞錯誤的參數的返回情況。
#./go_flag -a=b flag provided but not defined: -a cli : go_flag -port=8080 -config=/usr/local/service/config -logfile=/usr/local/service/log/service.log-config string配置文件配置參數為:-config=/usr/local/service/config-logfile string日志文件配置參數為:-logfile=/usr/local/service/log/service.log-port int服務端口設置參數為:-port=80-tcp_port intTCP服務端口描述: -tcp_port=2001跟蹤到代碼中可以看到如圖:
從上例可見,其實flag并沒有非常強大,下一章節會介紹一個更強大的命令行解析工具。
總結:
- os標準庫提供的解析方法,能夠解析簡單的命令行參數。
- flag能夠解析約定好的常規按照-flag傳遞的命令行參數,有幫助信息。
- os和flag還不能夠解析復雜結構的啟動參數。
?
總結
以上是生活随笔為你收集整理的Go语言 命令行解析(一)的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: Go语言 可变参数
- 下一篇: Go语言 命令行解析(二)