大家都說好用的 Python 命令列庫:click
作者:HelloGitHub-Prodesire
HelloGitHub 的《講解開源專案》系列,專案地址:https://github.com/HelloGitHub-Team/Article
一、前言
在本系列前面幾篇文章中,我們分別介紹了 argparse
和 docopt
的主要功能和用法。它們各具特色,都能出色地完成命令列任務。argparse
是面向過程的,需要先設定解析器,再定義引數,再解析命令列,最後實現業務邏輯。而 docopt
先用宣告式的語法定義出引數,再過程式地解析命令列和實現業務邏輯。在一些人看來,這些方式都不夠優雅。
而今天要介紹的 click 則是用一種你很熟知的方式來玩轉命令列。命令列程式本質上是定義引數和處理引數,而處理引數的邏輯一定是與所定義的引數有關聯的。那可不可以用函式和裝飾器來實現處理引數邏輯與定義引數的關聯呢?而 click
本系列文章預設使用 Python 3 作為直譯器進行講解。
若你仍在使用 Python 2,請注意兩者之間語法和庫的使用差異哦~
二、介紹
click 是一個以儘可能少的程式碼、以組合的方式建立優美的命令列程式的 Python 包。它有很高的可配置性,同時也能開箱即用。
它旨在讓編寫命令列工具的過程既快速又有趣,還能防止由於無法實現預期的 CLI API 所產生挫敗感。它有如下三個特點:
- 任意巢狀命令
- 自動生成幫助
- 支援執行時延遲載入子命令
三、快速開始
3.1 業務邏輯
首先定義業務邏輯,是不是感覺到有些難以置信呢?
不論是 argparse
還是 docopt
click
卻是放在第一步。細想想 click
的這種方式才更符合人的思維吧?不論用什麼命令列框架,我們最終關心的就是實現業務邏輯,其它的能省則省。
我們以官方示例為例,來介紹 click
的用法和哲學。假設命令列程式的輸入是 name
和 count
,功能是列印指定次數的名字。
那麼在 hello.py
中,很容易寫出如下程式碼:
def hello(count, name): """Simple program that greets NAME for a total of COUNT times.""" for x in range(count): click.echo('Hello %s!' % name)
這段程式碼的邏輯很簡單,就是迴圈 count
次,使用 click.echo
列印 name
。其中,click.echo
和 print
的作用相似,但功能更加強大,能處理好 Unicode 和 二進位制資料的情況。
3.2 定義引數
很顯然,我們需要針對 count
和 name
來定義它們所對應的引數資訊。
count
對應為命令列選項--count
,型別為數字,我們希望在不提供引數時,其預設值是 1name
對應為命令列選項--name
,型別為字串,我們希望在不提供引數時,能給人提示
使用 click
,就可以寫成下面這樣:
from click import click
@click.command()
@click.option('--count', default=1, help='Number of greetings.')
@click.option('--name', prompt='Your name',
help='The person to greet.')
def hello(count, name):
...
在上面的示例中:
- 使用裝飾器的方式,即定義了引數,又將之與處理邏輯繫結,這真是優雅。和
argparse
、docopt
比起來,就少了一步繫結過程 - 使用
click.command
表示hello
是對命令的處理 - 使用
click.option
來定義引數選項- 對於
--count
來說,使用default
來指定預設值。而由於預設值是數字,進而暗示--count
選項的型別為數字 - 對於
--name
來說,使用prompt
來指定未輸入該選項時的提示語 - 使用
help
來指定幫助資訊
- 對於
不論是裝飾器的方式、還是各種預設行為,click
都是像它的介紹所說的那樣,讓人儘可能少地編寫程式碼,讓整個過程變得快速而有趣。
3.3 程式碼梳理
使用 click
的方式非常簡單,我們將上文的程式碼彙總下,以有一個更清晰的認識:
# hello.py
import click
@click.command()
@click.option('--count', default=1, help='Number of greetings.')
@click.option('--name', prompt='Your name',
help='The person to greet.')
def hello(count, name):
"""Simple program that greets NAME for a total of COUNT times."""
for x in range(count):
click.echo('Hello %s!' % name)
if __name__ == '__main__':
hello()
若我們指定次數和名字:
$ python3 hello.py --count 2 --name Eric
Hello Eric!
Hello Eric!
若我們什麼都不指定,則會提示輸入名字,並預設輸出一次:
$ python3 hello.py
Your name: Eric
Hello Eric!
我們還可以通過 --help
引數檢視自動生成的幫助資訊:
Usage: hello.py [OPTIONS]
Simple program that greets NAME for a total of COUNT times.
Options:
--count INTEGER Number of greetings.
--name TEXT The person to greet.
--help Show this message and exit.
四、小結
click
的思路非常簡單,定義處理函式,通過它的裝飾器來定義引數。使用裝飾器的絕妙之處就在於把定義和繫結這兩個步驟合為一個步驟,使得整個過程變得如絲般順滑。
click
除了以 Pythonic
的方式讓命令列程式的實現變得更加優雅和好用外,還提供了比 argparse
和 docopt
都要強大的功能。在接下來幾節中,我們將會逐步揭開它的面紗。
『講解開源專案系列』——讓對開源專案感興趣的人不再畏懼、讓開源專案的發起者不再孤單。跟著我們的文章,你會發現程式設計的樂趣、使用和發現參與開源專案如此簡單。歡迎留言聯絡我們、加入我們,讓更多人愛上開源、貢獻開源