一、前言

在近半年的 Python 命令行旅程中,我们依次学习了 argparse 、 docopt 、 click 和 fire 库的特点和用法,逐步了解到 Python 命令行库的设计哲学与演变。本文作为本次旅程的终点,希望从一个更高的视角对这些库进行横向对比,总结它们的异同点和使用场景,以期在应对不同场景时能够分析利弊,选择合适的库为己所用。

本系列文章默认使用 Python 3 作为解释器进行讲解。若你仍在使用 Python 2,请注意两者之间语法和库的使用差异哦~

二、设计理念

在讨论各个库的设计理念之前,我们先设计一个 计算器程序 ,其实这个例子在 argparse 库的第一篇讲解中出现过,也就是:

命令行程序接受一个位置参数,它能出现多次,且是数字

默认情况下,命令行程序会求出给定的一串数字的最大值

如果指定了选项参数 --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=, nums=[1, 2, 3])

# 4. 业务逻辑

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] ...

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, '': ['1', '2', '3']}

# 3. 业务逻辑

nums = (int(num) for num in arguments[''])

if arguments['--sum']:

result = sum(nums)

else:

result = max(nums)

print(result) # 基于上文的 ['--sum', '1', '2', '3'] 参数,处理函数为 sum 函数,其结果为 6

从上述示例可以看到,我们通过 __doc__ 定义了接口描述,这和 argparse 中 add_argument 是等价的,然后 docopt 便会根据这个元信息把命令行参数转换为一个字典。业务逻辑中就需要对这个字典进行处理。

对比与 argparse :

对于更为复杂的命令程序,元信息的定义上 docopt 会更加简单

然而在业务逻辑的处理上,由于 argparse 在一些简单参数的处理上会更加便捷(比如示例中的情形),相对来说 docopt 转换为字典后就把所有处理交给业务逻辑的方式会更加复杂

2.3、click

命令行程序本质上是定义参数和处理参数,而处理参数的逻辑一定是与所定义的参数有关联的。那可不可以用函数和装饰器来实现处理参数逻辑与定义参数的关联呢?而 click 正好就是以这种使用方式来设计的。

click 使用装饰器的好处就在于用装饰器优雅的语法将参数定义和处理逻辑整合在一起,从而暗示了路由关系。相比于 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. 业务逻辑

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 最明显的优势。

此外, click 还内置了很多实用工具和额外能力,比如说 Bash 补全、颜色、分页支持、进度条等诸多实用功能,可谓是如虎添翼。

2.4、fire

fire 则是用一种面向广义对象的方式来玩转命令行,这种对象可以是类、函数、字典、列表等,它更加灵活,也更加简单。你都不需要定义参数类型, fire 会根据输入和参数默认值来自动判断,这无疑进一步简化了实现过程。

以下示例为 fire 实现的 计算器程序 :

import sys

import fire

sys.argv = ['calculator.py', '1', '2', '3', '--sum']

builtin_sum = sum

# 1. 业务逻辑

# 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 的了。我们只需关注业务逻辑,而命令行参数的定义则和函数参数的定义融为了一体。

不过,有利自然也有弊,比如 nums 并没有说是什么类型,也就意味着输入字符串'abc'也是合法的,这就意味着一个严格的命令行程序必须在自己的业务逻辑中来对期望的类型进行约束。

三、横向对比

最后,我们横向对比下 argparse 、 docopt 、 click 和 fire 库的各项功能和特点:

Python 的命令行库种类繁多、各具特色。结合上面的总结,可以选择出符合使用场景的库,如果几个库都符合,那么就根据你更偏爱的风格来选择。这些库都很优秀,其背后的思想很是值得我们学习和扩展。

总结

以上所述是小编给大家介绍的用什么库写 Python 命令行程序,希望对大家有所帮助,也非常感谢大家对脚本之家网站的支持!

python命令行大全-用什么库写 Python 命令行程序(示例代码详解)相关推荐

  1. python 代码命令大全-用什么库写 Python 命令行程序(示例代码详解)

    一.前言 在近半年的 Python 命令行旅程中,我们依次学习了 argparse . docopt . click 和 fire 库的特点和用法,逐步了解到 Python 命令行库的设计哲学与演变. ...

  2. python什么模块动态调用链接库_Python调用C/C++动态链接库的方法详解

    本文以实例讲解了Python调用C/C++ DLL动态链接库的方法,具体示例如下: 示例一: 首先,在创建一个DLL工程(本例创建环境为VS 2005),头文件: 1 2 3 4 5 6 7 8 9 ...

  3. python通过DictReader实现两个csv文件的映射查找lookup之代码详解

    python里应用DictReader的实现匹配查找 需求简介 通过应用DictReader读取csv文件,实现通过一个CSV在另外个CSV里找到对应的解释,类似lookup. 数据内容 1 这里假定 ...

  4. PyKivy入门教程:Kivy Python库的介绍、安装及使用方法详解

    PyKivy入门教程:Kivy Python库的介绍.安装及使用方法详解 Kivy是一个跨平台的Python GUI工具包,可以用于开发能够在多种设备和操作系统上运行的应用程序.Kivy支持触摸输入. ...

  5. python代码大全表解释-python操作列表的函数使用代码详解

    python的列表很重要,学习到后面你会发现使用的地方真的太多了.最近在写一些小项目时经常用到列表,有时其中的方法还会忘哎! 所以为了复习写下了这篇博客,大家也可以来学习一下,应该比较全面和详细了 列 ...

  6. 从头开始写STM32F103C8T6驱动库(四)——编写延时函数,详解Systick

    系列文章目录 Github开源地址 从头开始写STM32F103C8T6驱动库(一)--STM32CubeMX创建并调整工程结构 从头开始写STM32F103C8T6驱动库(二)--编写系统初始化程序 ...

  7. python split函数 空格_最易懂的Python新手教程:从基础语法到代码详解

    导读:本文立足基础,讲解Python和PyCharm的安装,及Python最简单的语法基础和爬虫技术中所需的Python语法. 作者:罗攀 蒋仟 如需转载请联系华章科技 本文涉及的主要知识点如下: P ...

  8. python类装饰器详解-Python类中的装饰器在当前类中的声明与调用详解

    我的Python环境:3.7 在Python类里声明一个装饰器,并在这个类里调用这个装饰器. 代码如下: class Test(): xx = False def __init__(self): pa ...

  9. 基于python的随机森林回归实现_PYTHON | 随机森林实战(代码+详解)

    大家好,我是菜鸟君,之前跟大家聊过R语言的随机森林建模,指路 R语言 | 随机森林建模实战(代码+详解),作为刚过完1024节日的码农算法工程师来说,怎么可能只会用一种语言呢?今天就来说说Python ...

最新文章

  1. Oracle高级查询之OVER (PARTITION BY ..)
  2. java面试题十五 for循环一个题目
  3. 继裁员15%后 滴滴宣布本月25日开始减少员工福利
  4. go bson转换成json_你会不会用Go语言实现redis的延时消息队列
  5. 08-求解Ax=b:可解性和解的结构
  6. matlab上机操作作业指导书,LED自动固晶基本操作1
  7. 在思科路由器上配置SSH登录
  8. ms sqlserver对象、所属用户、所属架构、登陆用户、同义词
  9. python基础教程ppt下载_python基础分享ppt
  10. WordPress 速度优化的完整指南
  11. 静态资源文件无法加载导致jsp页面渲染不成功问题
  12. Android 仿微信语音聊天,正式加入字节跳动
  13. 【小学信息技术教资面试】教案模板
  14. 寻根究底,探讨 chi -square特征词选择方法后面的数学支持
  15. Solr查询参数wt
  16. MSP430晶振配置详解
  17. mysql的事件开启,停止和启动
  18. 2020.9.28-10.5 人工智能行业每周技术精华文章汇总
  19. 安卓调用系统截图方法
  20. msyql 5.7.21 [ 35%] Building CXX object sql/CMakeFiles/sql.dir/item_row.cc.o

热门文章

  1. 【uoj#164】[清华集训2015]V 线段树维护历史最值
  2. spinner 下拉框控件
  3. sort (STL)
  4. 怎样解决xcode里开发cocos2dx改动lua脚本后不刷新的问题
  5. usaco2.1.2(frac1)
  6. Mac下布置appium环境
  7. 【oracle】复合数据类型
  8. Ex 5_33 实现一个关于公式长度(其中所有文字总的出现次数)为线性时间的Horn公式可满足性问题_第十次作业...
  9. WebAssembly 浏览器中运行c/c++模块
  10. java 实现 DES加密 解密算法