利用Python內(nèi)置庫實(shí)現(xiàn)創(chuàng)建命令行應(yīng)用程序
介紹
當(dāng)創(chuàng)建一個(gè)應(yīng)用程序時(shí),通常希望能夠告訴你的應(yīng)用程序如何做某事。有兩種流行的方法來完成這項(xiàng)任務(wù),你可以讓應(yīng)用程序接受命令行參數(shù),或者創(chuàng)建一個(gè)圖形化的用戶接口。有些應(yīng)用程序兩者都支持。
當(dāng)你需要在服務(wù)器上運(yùn)行你的代碼時(shí),命令行接口很有幫助。大多數(shù)服務(wù)器沒有圖形化界面,特別當(dāng)它們是Linux服務(wù)器時(shí)。在這種情況下,即使你想運(yùn)行圖形用戶界面,你也可能無法運(yùn)行。
Python 有一個(gè)叫做 argparse 的內(nèi)置庫,可以用它來創(chuàng)建一個(gè)命令行界面。在這篇文章中,你將了解到以下內(nèi)容。
- 解析實(shí)參
- 創(chuàng)建有用的信息
- 添加別名
- 使用相互排斥的參數(shù)
- 創(chuàng)建一個(gè)簡單的搜索工具
argparse模塊的內(nèi)容比本文要介紹的多得多。如果你想知道更多關(guān)于它的信息,你可以查看文檔。
現(xiàn)在是時(shí)候開始從命令行中解析參數(shù)了。
解析參數(shù)
在學(xué)習(xí)如何使用 argparse 之前,最好知道還有一種方法可以向 Python 腳本傳遞參數(shù)。你可以向 Python 腳本傳遞任何參數(shù),并通過使用 sys 模塊訪問這些參數(shù)。
為了了解它是如何工作的,創(chuàng)建一個(gè)名為 sys_args.py 的文件,并在其中輸入以下代碼:
# sys_args.py import sys def main(): print('You passed the following arguments:') print(sys.argv) if __name__ == '__main__': main()
這段代碼導(dǎo)入sys并打印出sys.argv中的任何內(nèi)容。argv屬性包含了所有傳遞給腳本的東西的列表,其中第一項(xiàng)是腳本本身。
下面是一個(gè)例子,說明當(dāng)你運(yùn)行這段代碼和幾個(gè)參數(shù)的時(shí)候會發(fā)生什么。
$ python3 sys_args.py --s 45 You passed the following arguments: ['sys_args.py', '--s', '45']
使用sys.argv的問題是,你無法控制可以傳遞給你的應(yīng)用程序的參數(shù)。
- 不能忽略參數(shù)
- 不能創(chuàng)建默認(rèn)參數(shù)
- 無法判斷什么是有效參數(shù)
這就是為什么使用 argparse 是使用 Python 標(biāo)準(zhǔn)庫的方法。argparse 模塊是非常強(qiáng)大和有用的。讓我們想一想,一個(gè)命令行應(yīng)用程序所遵循的常見過程。
- pass:傳入一個(gè)文件
- do:在你的程序中對該文件做一些處理
- output :輸出結(jié)果
下面是一個(gè)關(guān)于如何工作的通用例子。繼續(xù)創(chuàng)建file_parser.py并添加以下代碼。
# file_parser.py import argparse def file_parser(input_file, output_file=''): print(f'Processing {input_file}') print('Finished processing') if output_file: print(f'Creating {output_file}') def main(): parser = argparse.ArgumentParser('File parser') parser.add_argument('--infile', help='Input file') parser.add_argument('--out', help='Output file') args = parser.parse_args() if args.infile: file_parser(args.infile, args.out) if __name__ == '__main__': main()
file_parser()函數(shù)是進(jìn)行解析的邏輯所在。在這個(gè)例子中,它只接收一個(gè)文件名,并將其打印出來。output_file參數(shù)的默認(rèn)值是一個(gè)空字符串。
程序的重點(diǎn)在 main()中。在這里你創(chuàng)建了一個(gè) argparse.ArgumentParser()的實(shí)例,并給你的解析器起了一個(gè)名字。然后你添加兩個(gè)參數(shù), --infile和 --out。為了使用這個(gè)解析器,你需要調(diào)用 parse_args(),它將返回傳遞給你的程序的任何有效參數(shù)。最后,你要檢查用戶是否使用了 --infile 標(biāo)志。如果他們使用了,那么你就運(yùn)行 file_parser()。
下面是你如何在你的終端中運(yùn)行代碼。
$ python file_parser.py --infile something.txt Processing something.txt Finished processing
在這里,你用 --infile標(biāo)志和一個(gè)文件名來運(yùn)行你的腳本。這將運(yùn)行 main(),然后調(diào)用 file_parser()。
下一步是使用你在代碼中聲明的兩個(gè)命令行參數(shù)嘗試運(yùn)行你的應(yīng)用程序。
$ python file_parser.py --infile something.txt --out output.txt Processing something.txt Finished processing Creating output.txt
這一次,你得到了一個(gè)額外的輸出行,提到了輸出文件名。這代表你的代碼邏輯中的一個(gè)分支。當(dāng)你指定一個(gè)輸出文件時(shí),你可以讓你的代碼通過使用一個(gè)新的代碼塊或一個(gè)函數(shù)來生成該文件。如果你不指定一個(gè)輸出文件,那么那個(gè)代碼塊就不會運(yùn)行。
當(dāng)你使用argparse創(chuàng)建你的命令行工具時(shí),你可以很容易地添加信息,當(dāng)你的用戶不確定如何正確地與你的程序互動時(shí),可以幫助他們。
現(xiàn)在是時(shí)候找出如何從你的應(yīng)用程序中獲得幫助了
創(chuàng)建幫助信息
argparse庫將使用你在創(chuàng)建每個(gè)參數(shù)時(shí)提供的信息,自動為你的應(yīng)用程序創(chuàng)建一個(gè)有用的信息。這里是代碼:
# file_parser.py import argparse def file_parser(input_file, output_file=''): print(f'Processing {input_file}') print('Finished processing') if output_file: print(f'Creating {output_file}') def main(): parser = argparse.ArgumentParser('File parser') parser.add_argument('--infile', help='Input file') parser.add_argument('--out', help='Output file') args = parser.parse_args() if args.infile: file_parser(args.infile, args.out) if __name__ == '__main__': main()
現(xiàn)在試著用 -h標(biāo)志運(yùn)行這段代碼,你應(yīng)該看到以下內(nèi)容。
$ file_parser.py -h usage: File parser [-h] [--infile INFILE] [--out OUT] optional arguments: -h, --help show this help message and exit --infile INFILE Input file --out OUT Output file
add_argument()的幫助參數(shù)被用來創(chuàng)建上面的幫助信息。argparse會自動添加 -h和 -help選項(xiàng)。你可以通過給它一個(gè)描述和后記來使你的幫助信息更豐富。
讓我們用它們來改進(jìn)你的幫助信息。首先,把上面的代碼復(fù)制到一個(gè)新的文件中,命名為 file_parser_with_description.py,然后把它修改成下面的樣子。
# file_parser_with_description.py import argparse def file_parser(input_file, output_file=''): print(f'Processing {input_file}') print('Finished processing') if output_file: print(f'Creating {output_file}') def main(): parser = argparse.ArgumentParser( 'File parser', description='PyParse - The File Processor', epilog='Thank you for choosing PyParse!', ) parser.add_argument('--infile', help='Input file for conversion') parser.add_argument('--out', help='Converted output file') args = parser.parse_args() if args.infile: file_parser(args.infile, args.out) if __name__ == '__main__': main()
在這里,把description和epilog參數(shù)傳遞給ArgumentParser。還更新了 add_argument()的幫助參數(shù),使其更具描述性。
在做了這些修改之后,當(dāng)你用 -h或 --help運(yùn)行這個(gè)腳本時(shí),你會看到以下輸出。
$ python file_parser_with_description.py -h usage: File parser [-h] [--infile INFILE] [--out OUT] PyParse - The File Processor optional arguments: -h, --help show this help message and exit --infile INFILE Input file for conversion --out OUT Converted output file Thank you for choosing PyParse!
現(xiàn)在可以在你的幫助輸出中看到新的description 和epilog。這給了你的命令行程序一些額外的修飾。
你也可以通過ArgumentParser的 add_help參數(shù)在你的應(yīng)用程序中完全禁用幫助。如果你認(rèn)為你的幫助文本過于冗長,你可以像這樣禁用它。
# file_parser_no_help.py import argparse def file_parser(input_file, output_file=''): print(f'Processing {input_file}') print('Finished processing') if output_file: print(f'Creating {output_file}') def main(): parser = argparse.ArgumentParser( 'File parser', description='PyParse - The File Processor', epilog='Thank you for choosing PyParse!', add_help=False, ) parser.add_argument('--infile', help='Input file for conversion') parser.add_argument('--out', help='Converted output file') args = parser.parse_args() if args.infile: file_parser(args.infile, args.out) if __name__ == '__main__': main()
通過將 add_help設(shè)置為 False,你將禁用 -h和 --help標(biāo)志。
你可以看到下面的演示。
$ python file_parser_no_help.py --help usage: File parser [--infile INFILE] [--out OUT] File parser: error: unrecognized arguments: --help
在下一節(jié)中,你將學(xué)習(xí)如何為你的參數(shù)添加別名!
添加別名
別名是一個(gè)花哨的詞,指的是使用一個(gè)替代的標(biāo)志來做同樣的事情。例如,你知道你可以使用 -h和 --help來訪問程序的幫助信息。-h是 --help的別名,反之亦然。
看看 main()里面的 parser.add_argument()方法有什么變化。
# file_parser_aliases.py import argparse def file_parser(input_file, output_file=''): print(f'Processing {input_file}') print('Finished processing') if output_file: print(f'Creating {output_file}') def main(): parser = argparse.ArgumentParser( 'File parser', description='PyParse - The File Processor', epilog='Thank you for choosing PyParse!', add_help=False, ) parser.add_argument('-i', '--infile', help='Input file for conversion') parser.add_argument('-o', '--out', help='Converted output file') args = parser.parse_args() if args.infile: file_parser(args.infile, args.out) if __name__ == '__main__': main()
這里你改變了第一個(gè) add_argument(),除了接受 -infile之外,還接受了 -i,你還在第二個(gè) add_argument()中加入了 -o。這樣就可以使用兩個(gè)新的快捷標(biāo)志來運(yùn)行你的代碼。
下面是一個(gè)例子。
$ python3 file_parser_aliases.py -i something.txt -o output.txt Processing something.txt Finished processing Creating output.txt
如果你去看argparse文檔,你會發(fā)現(xiàn)也可以給子解析器添加別名。子解析器是一種在你的應(yīng)用程序中創(chuàng)建子命令的方法,這樣它就可以做其他事情。一個(gè)很好的例子是Docker,一個(gè)虛擬化或容器應(yīng)用程序。它有一系列的命令,你可以在docker下運(yùn)行,以及docker compose等等。這些命令中的每一個(gè)都有獨(dú)立的子命令,你都可以使用。
下面是一個(gè)典型的docker命令,用于運(yùn)行一個(gè)容器。
docker exec -it container_name bash
這將用docker啟動一個(gè)容器。而如果你要使用docker compose,你將使用一組不同的命令。exec和compose是subparsers的例子。
使用相互排斥的參數(shù)
有時(shí)你需要讓你的應(yīng)用程序接受一些參數(shù),但不接受其他參數(shù)。例如,你可能想限制你的應(yīng)用程序,使它只能創(chuàng)建或刪除文件,而不是同時(shí)創(chuàng)建和刪除。
argparse模塊提供了 add_mutually_exclusive_group()方法,它就是這樣做的。
將你的兩個(gè)參數(shù)添加到一個(gè)組對象中,使其相互排斥,如下面的例子。
# file_parser_exclusive.py import argparse def file_parser(input_file, output_file=''): print(f'Processing {input_file}') print('Finished processing') if output_file: print(f'Creating {output_file}') def main(): parser = argparse.ArgumentParser( 'File parser', description='PyParse - The File Processor', epilog='Thank you for choosing PyParse!', add_help=False, ) group = parser.add_mutually_exclusive_group() group.add_argument('-i', '--infile', help='Input file for conversion') group.add_argument('-o', '--out', help='Converted output file') args = parser.parse_args() if args.infile: file_parser(args.infile, args.out) if __name__ == '__main__': main()
首先,你創(chuàng)建了一個(gè)相互排斥的組。然后,你把 -i和 -o參數(shù)添加到組中,而不是添加到解析器對象中?,F(xiàn)在這兩個(gè)參數(shù)是互斥的。
下面是當(dāng)你試圖用這兩個(gè)參數(shù)運(yùn)行你的代碼時(shí)發(fā)生的情況。
$ python3 file_parser_exclusive.py -i something.txt -o output.txt usage: File parser [-i INFILE | -o OUT] File parser: error: argument -o/--out: not allowed with argument -i/--infile
用這兩個(gè)參數(shù)運(yùn)行你的代碼,會使你的解析器向用戶顯示一條錯(cuò)誤信息,解釋他們做錯(cuò)了什么。
在涵蓋了所有這些與使用argparse有關(guān)的信息之后,你已經(jīng)準(zhǔn)備好應(yīng)用你的新技能來創(chuàng)建一個(gè)簡單的搜索工具了
創(chuàng)建一個(gè)簡單的搜索工具
在開始創(chuàng)建一個(gè)應(yīng)用程序之前,弄清楚你要完成的任務(wù)總是好的。你在本節(jié)中想要建立的應(yīng)用程序應(yīng)該能夠搜索特定文件類型的文件。為了使它更有趣,你可以添加一個(gè)額外的參數(shù),讓你也能選擇性地搜索特定的文件大小。
你可以使用 Python 的 glob 模塊來搜索文件類型。你可以在這里閱讀關(guān)于這個(gè)模塊的所有信息。
還有一個(gè) fnmatch 模塊,glob 自己也使用它。你現(xiàn)在應(yīng)該使用 glob,因?yàn)樗菀资褂?,但是如果你有興趣寫一些更專業(yè)的東西,那么 fnmatch 可能是你正在尋找的。
然而,由于你希望能夠通過文件大小來選擇性地過濾返回的文件,你可以使用 pathlib,它包括一個(gè)類似 glob 的接口。glob 模塊本身并不提供文件大小的信息。
你可以先創(chuàng)建一個(gè)名為 pysearch.py 的文件并輸入以下代碼。
# pysearch.py import argparse import pathlib def search_folder(path, extension, file_size=None): """ Search folder for files """ folder = pathlib.Path(path) files = list(folder.rglob(f'*.{extension}')) if not files: print(f'No files found with {extension=}') return if file_size is not None: files = [ f for f in files if f.stat().st_size >= file_size ] print(f'{len(files)} *.{extension} files found:') for file_path in files: print(file_path)
在上面的代碼片段中,首先導(dǎo)入了argparse和pathlib。接下來,創(chuàng)建了search_folder()函數(shù),它接收了三個(gè)參數(shù)。
- path - 要搜索的文件夾
- extension - 要尋找的文件擴(kuò)展名
- file_size - 要過濾的文件大小,以字節(jié)為單位。
把路徑變成pathlib.Path對象,然后使用其rglob()方法在文件夾中搜索用戶傳入的擴(kuò)展名。如果沒有找到文件,就向用戶打印一個(gè)有意義的信息,然后退出。
如果找到了任何文件,就檢查是否已經(jīng)設(shè)置了filesize。如果它被設(shè)置了,就用一個(gè)list comprehension來過濾出小于指定的filesize的文件。
接下來,打印出找到的文件的數(shù)量,最后在這些文件上循環(huán),打印出它們的名字。
為了使這一切正常工作,需要創(chuàng)建一個(gè)命令行界面。你可以通過添加一個(gè)包含argparse代碼的main()函數(shù)來做到這一點(diǎn),像這樣。
def main(): parser = argparse.ArgumentParser( 'PySearch', description='PySearch - The Python Powered File Searcher', ) parser.add_argument('-p', '--path', help='The path to search for files', required=True, dest='path') parser.add_argument('-e', '--ext', help='The extension to search for', required=True, dest='extension') parser.add_argument('-s', '--size', help='The file size to filter on in bytes', type=int, dest='size', default=None) args = parser.parse_args() search_folder(args.path, args.extension, args.size) if __name__ == '__main__': main()
這個(gè)ArgumentParser()有三個(gè)參數(shù),與你傳遞給search_folder()的參數(shù)相對應(yīng)。讓--path和--ext參數(shù)成為必需的,而讓--size參數(shù)成為可選的。注意,--size參數(shù)被設(shè)置為type=int,這意味著你不能把它傳成字符串。
add_argument()函數(shù)有一個(gè)新的參數(shù)。它是dest參數(shù),可以用它來告訴你的參數(shù)分析器在哪里保存?zhèn)鬟f給它們的參數(shù)。
下面是一個(gè)腳本運(yùn)行的例子。
$ python3 pysearch.py -p /Users/michael/Dropbox/python101code/chapter32_argparse -e py -s 650 6 *.py files found: /Users/michael/Dropbox/python101code/chapter32_argparse/file_parser_aliases2.py /Users/michael/Dropbox/python101code/chapter32_argparse/pysearch.py /Users/michael/Dropbox/python101code/chapter32_argparse/file_parser_aliases.py /Users/michael/Dropbox/python101code/chapter32_argparse/file_parser_with_description.py /Users/michael/Dropbox/python101code/chapter32_argparse/file_parser_exclusive.py /Users/michael/Dropbox/python101code/chapter32_argparse/file_parser_no_help.py
現(xiàn)在試試用-s和一個(gè)字符串來運(yùn)行它。
$ python3 pysearch.py -p /Users/michael/Dropbox/python101code/chapter32_argparse -e py -s python usage: PySearch [-h] -p PATH -e EXTENSION [-s SIZE] PySearch: error: argument -s/--size: invalid int value: 'python'
這次我們收到了一個(gè)錯(cuò)誤,因?yàn)?s和-size只接受整數(shù)。在你自己的機(jī)器上運(yùn)行一下這段代碼,看看當(dāng)你使用-s和整數(shù)時(shí),它是否按你想要的方式工作。
這里有一些想法,你可以用來改進(jìn)你的代碼版本。
更好地處理擴(kuò)展文件。現(xiàn)在,它將接受 *.py,這不會像你期望的那樣工作。
更新代碼,以便你可以一次搜索多個(gè)擴(kuò)展名
更新代碼,以便對文件大小的范圍進(jìn)行過濾(例如,1MB-5MB)。
還有很多其他的功能和改進(jìn),你可以添加到這個(gè)代碼中,比如添加錯(cuò)誤處理或單元測試。
總結(jié)
argparse模塊功能齊全,可以用來創(chuàng)建龐大、靈活的命令行應(yīng)用程序。在本章中,你了解了以下內(nèi)容。
- 解析參數(shù)
- 創(chuàng)建有用的信息
- 添加別名
- 使用相互排斥的參數(shù)
- 創(chuàng)建一個(gè)簡單的搜索工具
你可以用argparse模塊做更多的事情,不完全包括本章所講的。請務(wù)必查看文檔以了解全部細(xì)節(jié)。
到此這篇關(guān)于利用Python內(nèi)置庫實(shí)現(xiàn)創(chuàng)建命令行應(yīng)用程序的文章就介紹到這了,更多相關(guān)Python命令行應(yīng)用程序內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Tensorflow加載與預(yù)處理數(shù)據(jù)詳解實(shí)現(xiàn)方法
讀取大型數(shù)據(jù)集并對其進(jìn)行有效預(yù)處理可能對其他深度學(xué)習(xí)庫來說很難實(shí)現(xiàn),但是TensorFlow借助Data API很容易實(shí)現(xiàn):只需創(chuàng)建一個(gè)數(shù)據(jù)集對象,并告訴它如何從何處獲取數(shù)據(jù)以及如何對其進(jìn)行轉(zhuǎn)換2022-11-11Pandas數(shù)據(jù)清洗函數(shù)總結(jié)
本文主要介紹了Pandas數(shù)據(jù)清洗函數(shù)總結(jié),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2023-01-01講解Python的Scrapy爬蟲框架使用代理進(jìn)行采集的方法
這篇文章主要介紹了講解Python的Scrapy爬蟲框架使用代理進(jìn)行采集的方法,并介紹了隨機(jī)使用預(yù)先設(shè)好的user-agent來進(jìn)行爬取的用法,需要的朋友可以參考下2016-02-02Python的Flask框架中實(shí)現(xiàn)分頁功能的教程
這篇文章主要介紹了Python的Flask框架中實(shí)現(xiàn)分頁功能的教程,文中的示例基于一個(gè)博客來實(shí)現(xiàn),需要的朋友可以參考下2015-04-04Python實(shí)現(xiàn)控制臺中的進(jìn)度條功能代碼
下面小編就為大家分享一篇Python實(shí)現(xiàn)控制臺中的進(jìn)度條功能代碼,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧2017-12-12十一個(gè)案例帶你吃透Python函數(shù)參數(shù)
這篇文章主要通過十一個(gè)案例帶大家一起了解一下Python中的函數(shù)參數(shù),文中的示例代碼講解詳細(xì),對我們學(xué)習(xí)Python有一定幫助,需要的可以參考一下2022-08-08django處理select下拉表單實(shí)例(從model到前端到post到form)
這篇文章主要介紹了django處理select下拉表單實(shí)例(從model到前端到post到form),具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧2020-03-03Python中sorted()函數(shù)的強(qiáng)大排序技術(shù)實(shí)例探索
排序在編程中是一個(gè)基本且重要的操作,而Python的sorted()函數(shù)則為我們提供了強(qiáng)大的排序能力,在本篇文章中,我們將深入研究不同排序算法、sorted()?函數(shù)的靈活性,以及各種排序場景下的最佳實(shí)踐2024-01-01