亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

用什么庫寫 Python 命令行程序(示例代碼詳解)

 更新時間:2020年02月20日 11:58:07   作者:HelloGitHub  
這篇文章主要介紹了用什么庫寫 Python 命令行程序,這一篇文章給大家介紹的非常詳細全面看這一天足夠了,需要的朋友可以參考下

一、前言

在近半年的 Python 命令行旅程中,我們依次學習了 argparse 、 docopt 、 click 和 fire 庫的特點和用法,逐步了解到 Python 命令行庫的設計哲學與演變。本文作為本次旅程的終點,希望從一個更高的視角對這些庫進行橫向對比,總結它們的異同點和使用場景,以期在應對不同場景時能夠分析利弊,選擇合適的庫為己所用。

本系列文章默認使用 Python 3 作為解釋器進行講解。若你仍在使用 Python 2,請注意兩者之間語法和庫的使用差異哦~

二、設計理念

在討論各個庫的設計理念之前,我們先設計一個 計算器程序 ,其實這個例子在 argparse 庫的第一篇講解中出現過,也就是:

  1. 命令行程序接受一個位置參數,它能出現多次,且是數字
  2. 默認情況下,命令行程序會求出給定的一串數字的最大值
  3. 如果指定了選項參數 --sum ,那么就會將求出給定的一串數字的和

希望從各個庫實現該例子的代碼中能進一步體會它們的設計理念。

2.1、argparse

argparse 的設計理念就是提供給你最細粒度的控制,你需要詳細地告訴它參數是選項參數還是位置參數、參數值的類型是什么、該參數的處理動作是怎樣的??傊?,它就像是一個沒有智能分析能力的初代機器人,你需要告訴它明確的信息,它才會根據給定的信息去幫助你做事情。

以下示例為 argparse 實現的 計算器程序 :

import argparse
# 1. 設置解析器
parser = argparse.ArgumentParser(description='Calculator Program.')
# 2. 定義參數
# 添加位置參數 nums,在幫助信息中顯示為 num
# 其類型為 int,且支持輸入多個,且至少需要提供一個
parser.add_argument('nums', metavar='num', type=int, nargs='+',
   help='a num for the accumulator')
# 添加選項參數 --sum,該參數被 parser 解析后所對應的屬性名為 accumulate
# 若不提供 --sum,默認值為 max 函數,否則為 sum 函數
parser.add_argument('--sum', dest='accumulate', action='store_const',
   const=sum, default=max,
   help='sum the nums (default: find the max)')
# 3. 解析參數
args = parser.parse_args(['--sum', '1', '2', '3'])
print(args) # 結果:Namespace(accumulate=<built-in function sum>, nums=[1, 2, 3])
# 4. 業(yè)務邏輯
result = args.accumulate(args.nums)
print(result) # 基于上文的 ['--sum', '1', '2', '3'] 參數,accumulate 為 sum 函數,其結果為 6

從上述示例可以看到,我們需要通過 add_argument 很明確地告訴 argparse 參數長什么樣:

  • 它是位置參數 nums ,還是選項參數 --sum
  • 它的類型是什么,比如 type=int 表示類型是 int
  • 這個參數能重復出現幾次,比如 nargs='+' 表示至少提供 1 個
  • 參數的是存什么的,比如 action='store_const' 表示存常量

然后它才根據給定的這些元信息來解析命令行參數(也就是示例中的 ['--sum', '1', '2', '3'] )。

這是很計算機的思維,雖然冗長,但也帶來了靈活性。

2.2、docopt

從 argparse 的理念可以看出,它是命令式的。這時候 docopt 另辟蹊徑,聲明式是不是也可以?一個命令行程序的幫助信息其實已然包含了這個命令行的完整元信息,那不就可以通過定義幫助信息來定義命令行? docopt 就是基于這樣的想法去設計的。

聲明式的好處在于只要你掌握了聲明式的語法,那么定義命令行的元信息就會很簡單。

以下示例為 docopt 實現的 計算器程序 :

# 1. 定義接口描述/幫助信息
"""Calculator Program.
Usage:
 calculator.py [--sum] <num>...
 calculator.py (-h | --help)
Options:
 -h --help Show help.
 --sum  Sum the nums (default: find the max).
"""
from docopt import docopt
# 2. 解析命令行
arguments = docopt(__doc__, options_first=True, argv=['--sum', '1', '2', '3'])
print(arguments) # 結果:{'--help': False, '--sum': True, '<num>': ['1', '2', '3']}
# 3. 業(yè)務邏輯
nums = (int(num) for num in arguments['<num>'])
if arguments['--sum']:
 result = sum(nums)
else:
 result = max(nums)
print(result) # 基于上文的 ['--sum', '1', '2', '3'] 參數,處理函數為 sum 函數,其結果為 6

從上述示例可以看到,我們通過 __doc__ 定義了接口描述,這和 argparse 中 add_argument 是等價的,然后 docopt 便會根據這個元信息把命令行參數轉換為一個字典。業(yè)務邏輯中就需要對這個字典進行處理。

對比與 argparse :

  • 對于更為復雜的命令程序,元信息的定義上 docopt 會更加簡單
  • 然而在業(yè)務邏輯的處理上,由于 argparse 在一些簡單參數的處理上會更加便捷(比如示例中的情形),相對來說 docopt 轉換為字典后就把所有處理交給業(yè)務邏輯的方式會更加復雜

2.3、click

命令行程序本質上是定義參數和處理參數,而處理參數的邏輯一定是與所定義的參數有關聯的。那可不可以用函數和裝飾器來實現處理參數邏輯與定義參數的關聯呢?而 click 正好就是以這種使用方式來設計的。

click 使用裝飾器的好處就在于用裝飾器優(yōu)雅的語法將參數定義和處理邏輯整合在一起,從而暗示了路由關系。相比于 argparse 和 docopt 需要自行對解析后的參數來做路由關系,簡單了不少。

以下示例為 click 實現的 計算器程序 :

import sys
import click
sys.argv = ['calculator.py', '--sum', '1', '2', '3']
# 2. 定義參數
@click.command()
@click.argument('nums', nargs=-1, type=int)
@click.option('--sum', 'use_sum', is_flag=True, help='sum the nums (default: find the max)')
# 1. 業(yè)務邏輯
def calculator(nums, use_sum):
 """Calculator Program."""
 print(nums, use_sum) # 輸出:(1, 2, 3) True
 if use_sum:
 result = sum(nums)
 else:
 result = max(nums)
 print(result) # 基于上文的 ['--sum', '1', '2', '3'] 參數,處理函數為 sum 函數,其結果為 6
calculator()

從上述示例可以看出,參數和對應的處理邏輯非常好地綁定在了一起,看上去就很直觀,使得我們可以明確了解參數會怎么處理,這在有大量參數時顯得尤為重要,這邊是 click 相比于 argparse 和 docopt 最明顯的優(yōu)勢。

此外, click 還內置了很多實用工具和額外能力,比如說 Bash 補全、顏色、分頁支持、進度條等諸多實用功能,可謂是如虎添翼。

2.4、fire

fire 則是用一種面向廣義對象的方式來玩轉命令行,這種對象可以是類、函數、字典、列表等,它更加靈活,也更加簡單。你都不需要定義參數類型, fire 會根據輸入和參數默認值來自動判斷,這無疑進一步簡化了實現過程。

以下示例為 fire 實現的 計算器程序 :

import sys
import fire
sys.argv = ['calculator.py', '1', '2', '3', '--sum']
builtin_sum = sum
# 1. 業(yè)務邏輯
# sum=False,暗示它是一個選項參數 --sum,不提供的時候為 False
# *nums 暗示它是一個能提供任意數量的位置參數
def calculator(sum=False, *nums):
 """Calculator Program."""
 print(sum, nums) # 輸出:True (1, 2, 3)
 if sum:
 result = builtin_sum(nums)
 else:
 result = max(nums)
 print(result) # 基于上文的 ['1', '2', '3', '--sum'] 參數,處理函數為 sum 函數,其結果為 6
fire.Fire(calculator)

從上述示例可以看出, fire 提供的方式無疑是最簡單、并且最 Pythonic 的了。我們只需關注業(yè)務邏輯,而命令行參數的定義則和函數參數的定義融為了一體。

不過,有利自然也有弊,比如 nums 并沒有說是什么類型,也就意味著輸入字符串'abc'也是合法的,這就意味著一個嚴格的命令行程序必須在自己的業(yè)務邏輯中來對期望的類型進行約束。

三、橫向對比

最后,我們橫向對比下 argparse 、 docopt 、 click 和 fire 庫的各項功能和特點:

Python 的命令行庫種類繁多、各具特色。結合上面的總結,可以選擇出符合使用場景的庫,如果幾個庫都符合,那么就根據你更偏愛的風格來選擇。這些庫都很優(yōu)秀,其背后的思想很是值得我們學習和擴展。

總結

以上所述是小編給大家介紹的用什么庫寫 Python 命令行程序,希望對大家有所幫助,也非常感謝大家對腳本之家網站的支持!

相關文章

  • TensorFlow損失函數專題詳解

    TensorFlow損失函數專題詳解

    本篇文章主要介紹了TensorFlow損失函數專題詳解,小編覺得挺不錯的,現在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2018-04-04
  • 解決python3.6用cx_Oracle庫連接Oracle的問題

    解決python3.6用cx_Oracle庫連接Oracle的問題

    這篇文章主要介紹了解決python3.6用cx_Oracle庫連接Oracle的問題,本文給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2020-12-12
  • OPENAI?API?微調?GPT-3?的?Ada?模型

    OPENAI?API?微調?GPT-3?的?Ada?模型

    這篇文章主要為大家介紹了OPENAI?API?微調?GPT-3?的?Ada?模型使用示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2023-04-04
  • 教你用Python腳本快速為iOS10生成圖標和截屏

    教你用Python腳本快速為iOS10生成圖標和截屏

    這篇文章主要介紹了教你用Python快速為iOS10生成圖標和截屏的相關資料,非常不錯,具有參考借鑒價值,需要的朋友可以參考下
    2016-09-09
  • 從CentOS安裝完成到生成詞云python的實例

    從CentOS安裝完成到生成詞云python的實例

    下面小編就為大家分享一篇從CentOS安裝完成到生成詞云python的實例,具有很好的參考價值,希望對大家有所幫助
    2017-12-12
  • Flask數據庫遷移簡單介紹

    Flask數據庫遷移簡單介紹

    這篇文章主要為大家詳細介紹了Flask數據庫遷移簡單工作,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2017-10-10
  • Python 中的lambda函數介紹

    Python 中的lambda函數介紹

    Lambda函數,即Lambda 表達式(lambda expression),是一個匿名函數(不存在函數名的函數),這篇文章主要介紹了Python lambda函數的基礎知識,需要的朋友可以參考下
    2018-10-10
  • Python函數中的全局變量詳解

    Python函數中的全局變量詳解

    大家好,本篇文章主要講的是Python函數中的全局變量詳解,感興趣的同學趕快來看一看吧,對你有幫助的話記得收藏一下,方便下次瀏覽
    2022-01-01
  • opencv 閾值分割的具體使用

    opencv 閾值分割的具體使用

    這篇文章主要介紹了opencv 閾值分割的具體使用,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2020-07-07
  • 詳解如何利用pandas進行數據行轉列和列轉行

    詳解如何利用pandas進行數據行轉列和列轉行

    這篇文章主要為大家詳細介紹了如何利用pandas進行數據行轉列和列轉行,文中的示例代碼講解詳細,感興趣的小伙伴可以跟隨小編一起學習一下
    2007-02-02

最新評論