当前位置: 首页 > news >正文

面向对象的 CLI:使用 Fire 简化类和对象的方法暴露 (中英双语)

面向对象的 CLI:使用 Fire 简化类和对象的方法暴露

在传统的命令行工具开发中,argparse 是最常用的库之一,用于处理命令行参数和配置。它通常用于函数式编程,但在处理类和对象时,使用起来可能不如 Fire 方便。Fire 是一个自动生成命令行接口的库,不仅支持函数,还能够支持类和对象。它通过动态地将 Python 类的属性和方法暴露给命令行,极大地简化了命令行工具的开发过程。

本文将探讨 Fire 如何支持类和对象,并通过一个示例来展示它的优势。我们将详细讲解代码,并分析 Fire 如何实现这一功能。

Fire 如何支持类和对象?

在使用 Fire 时,我们可以将类及其方法直接暴露为命令行工具,而无需手动设置参数解析。Fire 会自动分析类的构造函数(__init__)和方法(def),并生成对应的命令行接口。

关键原理:

  • 自动映射:Fire 会扫描类的构造函数和方法,自动识别方法参数,并将它们映射为命令行参数。
  • 递归支持:如果类方法中还有其他类的实例,Fire 会递归地将这些类和方法也暴露给命令行。
  • 自动类型推断:Fire 会根据函数签名推断参数的类型,包括数字、布尔值和字符串等。

通过这种方式,Fire 可以非常方便地将类及其方法转化为命令行工具,并允许用户像调用普通命令行工具一样,传递参数给类的方法。

示例:使用 Fire 曝露类和对象

让我们通过一个简单的例子来演示 Fire 如何支持类和对象。

代码示例

import fireclass DatasetManager:def __init__(self, repo_id):self.repo_id = repo_iddef upload(self, version: str, overwrite=False):print(f"Uploading dataset version {version} to {self.repo_id}...")if overwrite:print(f"Overwriting existing version {version}.")else:print(f"Version {version} will be added without overwriting.")def list_tags(self):print(f"Listing tags for {self.repo_id}...")class CLI:def __init__(self, repo_id):self.manager = DatasetManager(repo_id)def upload(self, version: str, overwrite=False):self.manager.upload(version, overwrite)def list_tags(self):self.manager.list_tags()if __name__ == "__main__":fire.Fire(CLI)

代码解析

  1. DatasetManager 类

    • DatasetManager 类包含两个方法:uploadlist_tagsupload 方法用于上传数据集并可选择是否覆盖现有版本,而 list_tags 用于列出数据集的标签。
  2. CLI 类

    • CLI 类封装了 DatasetManager 类的实例,并将其方法暴露为命令行工具。用户可以通过 CLI 类的方法来操作 DatasetManager
  3. fire.Fire(CLI)

    • fire.Fire(CLI)CLI 类的所有方法变成命令行工具,CLI 的每个方法会成为一个命令。

运行命令行

假设我们保存了以上代码到一个文件 cli_tool.py,然后通过命令行运行:

  1. 上传数据集
$ python cli_tool.py upload v1.0 --overwrite
Uploading dataset version v1.0 to None...
Overwriting existing version v1.0.
  1. 列出标签
$ python cli_tool.py list_tags
Listing tags for None...

解析

  • 在命令行中,uploadlist_tags 方法分别映射为两个命令。用户只需要按照方法的签名提供相应的参数,Fire 会自动处理参数解析。
  • versionoverwrite 被视为 upload 方法的命令行参数,而 repo_idDatasetManager 类的一个实例属性,默认在创建时传入。

自动生成 CLI 的工作原理

Fire 的核心机制是在后台使用 Python 的反射功能来检查类及其方法。当调用 fire.Fire(CLI) 时,Fire 会:

  • 扫描 CLI 类中的所有方法。
  • 检查每个方法的参数类型,并生成相应的命令行选项。
  • 如果方法中使用了其他类的实例,Fire 会递归地处理这些实例,使得整个类层级都能通过命令行操作。

Fire 的这个特性使得开发者能够快速为现有的类和对象生成命令行工具,而不需要编写冗长的命令行参数解析代码。

总结

Fire 的强大之处在于它能够简化命令行工具的开发,特别是对于包含多个类和对象的复杂项目。通过自动识别类的方法和参数,Fire 提供了一个简洁而强大的方式,将类和对象暴露为命令行接口。这种方式不仅减少了开发者的工作量,还提高了代码的可读性和可维护性。

对于那些需要在命令行中操作多个类或方法的情况,Fire 无疑是一个更合适的选择,特别是在需要快速原型开发时,它的面向对象支持使得代码更加简洁高效。


Object-Oriented CLI: How Fire Supports Classes and Objects

When developing command-line interfaces (CLI) in Python, libraries like argparse are commonly used for parsing command-line arguments. However, when it comes to dealing with classes and objects, Fire, a Python library developed by Google, offers significant advantages over argparse. Unlike argparse, which requires manually setting up argument parsing for each function, Fire can automatically expose Python classes, methods, and attributes as command-line commands, making it a more flexible and convenient tool for building CLIs.

In this blog post, we’ll explore how Fire supports classes and objects, demonstrate it with a code example, and dive into the internal workings of how Fire makes this possible.

How Fire Supports Classes and Objects

Fire simplifies the process of exposing Python classes and their methods as command-line interfaces by dynamically creating a CLI based on the structure of the class. It recursively discovers class methods and instance variables, making them available as command-line arguments.

Key Principles:

  1. Automatic Mapping: Fire scans classes and methods, automatically turning them into CLI commands.
  2. Recursive Support: If a method requires another class instance, Fire automatically makes that class and its methods available as subcommands.
  3. Type Inference: Fire can infer argument types based on the function signature, including strings, integers, booleans, etc.

This dynamic generation of CLI commands makes it incredibly easy to turn an object-oriented class into a functional command-line tool.

Example: Using Fire with Classes and Objects

Let’s walk through a simple example where we use Fire to expose a class and its methods via the command line.

Code Example

import fireclass DatasetManager:def __init__(self, repo_id):self.repo_id = repo_iddef upload(self, version: str, overwrite=False):print(f"Uploading dataset version {version} to {self.repo_id}...")if overwrite:print(f"Overwriting existing version {version}.")else:print(f"Version {version} will be added without overwriting.")def list_tags(self):print(f"Listing tags for {self.repo_id}...")class CLI:def __init__(self, repo_id):self.manager = DatasetManager(repo_id)def upload(self, version: str, overwrite=False):self.manager.upload(version, overwrite)def list_tags(self):self.manager.list_tags()if __name__ == "__main__":fire.Fire(CLI)

Explanation of the Code

  1. DatasetManager Class:

    • Contains methods for uploading datasets (upload) and listing dataset tags (list_tags).
    • The upload method has parameters for version and overwrite.
  2. CLI Class:

    • Encapsulates an instance of DatasetManager and exposes its methods (upload and list_tags) to the command line.
  3. fire.Fire(CLI):

    • This line exposes all the methods of the CLI class to the command line. CLI methods become commands that can be executed with arguments.

Running the CLI

Assuming you saved the above code to a file named cli_tool.py, you can execute it from the command line like this:

  1. Upload Dataset:

    $ python cli_tool.py upload v1.0 --overwrite
    Uploading dataset version v1.0 to None...
    Overwriting existing version v1.0.
    
  2. List Tags:

    $ python cli_tool.py list_tags
    Listing tags for None...
    

What Happens Behind the Scenes

  • Fire automatically exposes the methods upload and list_tags as CLI commands.
  • It takes care of parsing the arguments (version, overwrite) and passing them to the corresponding method of DatasetManager.
  • The repo_id is passed during the initialization of the CLI class, making it accessible in DatasetManager.

How Fire Implements Object-Oriented CLI

Internally, Fire works by using Python’s reflection capabilities to inspect classes and methods. Here’s how it works step by step:

  1. Class Inspection: Fire inspects the class passed to fire.Fire(), discovering all methods and attributes.
  2. Argument Mapping: For each method, Fire maps function parameters to command-line arguments.
  3. Recursive Command Creation: If a method contains an object or class instance, Fire recursively creates commands for that class as well.
  4. Argument Parsing: Fire uses Python’s built-in argparse module behind the scenes to handle argument parsing, but it generates the required configuration automatically based on the class signature.

This mechanism allows you to expose complex class hierarchies and objects without manually writing out argument parsing code.

Conclusion

Fire offers an elegant and efficient way to create CLIs from Python classes and objects. By automating the argument mapping and recursion, Fire eliminates much of the boilerplate code required by traditional CLI libraries like argparse. This makes Fire an excellent choice when working with complex object-oriented structures or when you need to quickly prototype command-line tools.

Whether you’re building a simple script or a large application with multiple classes, Fire provides a seamless interface to expose your code to the command line. Its automatic argument parsing, support for recursion, and object-oriented design make it a powerful tool for any developer looking to create a flexible and maintainable CLI.


后记

2024年12月15日19点40分于上海,在GPT4o mini大模型辅助下完成。

相关文章:

面向对象的 CLI:使用 Fire 简化类和对象的方法暴露 (中英双语)

面向对象的 CLI:使用 Fire 简化类和对象的方法暴露 在传统的命令行工具开发中,argparse 是最常用的库之一,用于处理命令行参数和配置。它通常用于函数式编程,但在处理类和对象时,使用起来可能不如 Fire 方便。Fire 是…...

flutter控件buildDragTargetWidget详解

文章目录 1. DragTarget 的核心概念基本属性 2. 基本用法3. 使用 buildDragTargetWidget4. 常见场景5. 注意事项 buildDragTargetWidget 不是 Flutter 中的内置 API 或方法,但根据命名习惯,它很可能是您正在实现或使用的一个方法,用于在 Flut…...

使用webrtc-streamer查看实时监控

摄像头配置(海康摄像头为例) 摄像头视频编码应改成H264格式 webrtc-streamer下载 webrtc-streamer下载地址 下载后解压出来双击运行,端口默认8000 VUE2项目引入文件 在项目静态文件“public”中需引入两个js文件“webrtcstreamer.js”与“…...

【数据分享】2014-2024年我国POI兴趣点数据(免费获取/来源于OSM地图)

POI是Point of Interest的简称,意为“兴趣点”,是互联网电子地图中用于表示特定位置的地理实体的核心数据类型。POI通常用于标注具体地点,例如餐厅、商场、学校、医院、景点等。这些数据以点的形式呈现,并附带详细属性信息&#x…...

Leetcode 3389. Minimum Operations to Make Character Frequencies Equal

Leetcode 3389. Minimum Operations to Make Character Frequencies Equal 1. 解题思路2. 代码实现 题目链接:3389. Minimum Operations to Make Character Frequencies Equal 1. 解题思路 这一题从答题从test的结果来说来说做出的人很少,主要确实有些…...

Vite 与 Webpack 的区别

在前端开发中,构建工具是不可或缺的,Webpack 和 Vite 是当前最流行的选择之一。尽管它们的目标相似,但在实现方式和开发体验上却有显著差异。本文将探讨 Vite 和 Webpack 的主要区别,以便于根据项目需求选择合适的工具。 1. 构建…...

基于32单片机的RS485综合土壤传感器检测土壤PH、氮磷钾的使用(超详细)

1-3为RS485综合土壤传感器的基本内容 4-5为基于STM32F103C8T6单片机使用RS485传感器检测土壤PH、氮磷钾并显示在OLED显示屏的相关配置内容 注意:本篇文件讲解使用的是PH、氮磷钾四合一RS485综合土壤传感器,但里面的讲解内容适配市面上的所有多合一的RS…...

【从零开始入门unity游戏开发之——C#篇11】一个标准 C# 程序介绍、新的值类型——枚举

文章目录 一、一个标准 C# 程序1、文件名(Program.cs):2、 using 语句:3、命名空间(namespace)4、类(class):4、入口函数(Main 方法)5、程序运行流…...

vue 签名校验 md5 uuid

import CryptoJS from crypto-js import uuid from /utils/uuid import { SECRET_KEY } from /utils/config // 签名校验 const nonceStr uuid.uuid() const timestamp new Date().getTime() // const sign CryptoJS.MD5(nonceStr nonceStr &secretKey SECRET_KEY …...

CSS系列(16)-- 架构与模式详解

前端技术探索系列:CSS 架构与模式详解 🏗️ 致读者:探索 CSS 架构的艺术 👋 前端开发者们, 今天我们将深入探讨 CSS 架构与设计模式,学习如何构建可维护的样式系统。 CSS 架构方法论 🚀 OO…...

【go语言】reflect包与类型推断

reflect 包的核心概念 Go 中的反射涉及两个核心概念: Type:表示一个类型的结构体,reflect.Type 是类型的描述。Value:表示一个值的结构体,reflect.Value 是一个具体值的包装。 反射让我们能够动态地访问对象的类型和…...

3.python运算符

Python 提供了多种运算符,用于执行算术、比较、逻辑等各种操作。以下是 Python 中常见的运算符类型及其用法: 文章目录 1. 算术运算符2. 比较运算符3. 逻辑运算符4. 赋值运算符5. 位运算符6. 成员运算符7. 身份运算符8. 运算符优先级 1. 算术运算符 算…...

【竞技宝】CS2-上海major:spirit力克MOUZ niko梦碎

北京时间2024年12月15日,CS2上海major正在如火如荼的进行中,昨日迎来两场半决赛MOUZ对阵spirit以及FAZE对阵G2。Spirit和MOUZ和各自赢下了自己的选图之后,spirit双子星在图三抗住压力帮助队伍杀入决赛。而G2和FAZE的比赛中,FAZE依然延续上一场的火热手感完全压制了G2,G2的明星选…...

【Leetcode 每日一题】3266. K 次乘运算后的最终数组 II

问题背景 给你一个整数数组 n u m s nums nums,一个整数 k k k 和一个整数 m u l t i p l i e r multiplier multiplier。 你需要对 n u m s nums nums 执行 k k k 次操作,每次操作中: 找到 n u m s nums nums 中的 最小 值 x x x&a…...

etcd集群常见日志

1、节点失去领导者 {"level":"info","ts":"2024-05-07T01:54:04.948Z","logger":"raft","caller":"etcdserver/zap_raft.go:77","msg":"raft.node: 9afce9447872453 lost le…...

【漫话机器学习系列】005.神经网络的结构(architecture on the neural network)

神经网络(Neural Network)是一种模拟人脑神经系统的计算模型,由大量相互连接的神经元(节点)组成,广泛应用于深度学习和机器学习领域。以下是神经网络的基本结构及关键组成部分。 1. 神经网络的基本组成 一…...

基于 Couchbase 数据仓库元数据管理的可行性方案

在大数据体系中,元数据管理是数据治理的关键一环。以下是一套元数据管理的可行性方案,适合你的当前架构设计(基于 Couchbase 数据仓库)并支持高效管理数据的分层与结构。 1. 元数据管理的目标 统一数据管理:清晰描述 …...

SpringBoot:快速构建微服务应用

一、SpringBoot简介 什么是SpringBoot 是由Pivotal团队提供的快速开发框架。它基于Spring框架,可以用于快速构建微服务应用程序。SpringBoot提供了一种快速、便捷的方式来启动和配置一个基于Spring的应用程序,它封装了很多常用的配置,简化了开…...

汽车嵌入式软件构建高效技术团队的全面思考

在汽车嵌入式软件开发领域,构建一支高效的通用技术团队至关重要。这类团队负责为各种项目提供可复用、标准化的技术基石,从而提高开发效率、降低成本并确保产品质量。构建这样的团队需要从技术能力、角色分工、标准化与复用、流程管理与质量保证、工具和…...

【跨库查询、多库查询】.NET开源 ORM 框架 SqlSugar 系列

文章目录 一、跨库方式1:跨库导航二、手动跨库查询三、同服务器:自动查询跨库查询3.1 Mysql和SqlServer自动3.2 自动: PgSql跨Scheme查询3.3 其他库同服务器 四、跨服务器:自动跨库查询4.1 配置SqlServer dblink4.2 配置 Oracle dblink4.3 配…...

MongoDB学习和应用(高效的非关系型数据库)

一丶 MongoDB简介 对于社交类软件的功能,我们需要对它的功能特点进行分析: 数据量会随着用户数增大而增大读多写少价值较低非好友看不到其动态信息地理位置的查询… 针对以上特点进行分析各大存储工具: mysql:关系型数据库&am…...

Mybatis逆向工程,动态创建实体类、条件扩展类、Mapper接口、Mapper.xml映射文件

今天呢,博主的学习进度也是步入了Java Mybatis 框架,目前正在逐步杨帆旗航。 那么接下来就给大家出一期有关 Mybatis 逆向工程的教学,希望能对大家有所帮助,也特别欢迎大家指点不足之处,小生很乐意接受正确的建议&…...

Python爬虫实战:研究feedparser库相关技术

1. 引言 1.1 研究背景与意义 在当今信息爆炸的时代,互联网上存在着海量的信息资源。RSS(Really Simple Syndication)作为一种标准化的信息聚合技术,被广泛用于网站内容的发布和订阅。通过 RSS,用户可以方便地获取网站更新的内容,而无需频繁访问各个网站。 然而,互联网…...

新能源汽车智慧充电桩管理方案:新能源充电桩散热问题及消防安全监管方案

随着新能源汽车的快速普及,充电桩作为核心配套设施,其安全性与可靠性备受关注。然而,在高温、高负荷运行环境下,充电桩的散热问题与消防安全隐患日益凸显,成为制约行业发展的关键瓶颈。 如何通过智慧化管理手段优化散…...

AI+无人机如何守护濒危物种?YOLOv8实现95%精准识别

【导读】 野生动物监测在理解和保护生态系统中发挥着至关重要的作用。然而,传统的野生动物观察方法往往耗时耗力、成本高昂且范围有限。无人机的出现为野生动物监测提供了有前景的替代方案,能够实现大范围覆盖并远程采集数据。尽管具备这些优势&#xf…...

【C++特殊工具与技术】优化内存分配(一):C++中的内存分配

目录 一、C 内存的基本概念​ 1.1 内存的物理与逻辑结构​ 1.2 C 程序的内存区域划分​ 二、栈内存分配​ 2.1 栈内存的特点​ 2.2 栈内存分配示例​ 三、堆内存分配​ 3.1 new和delete操作符​ 4.2 内存泄漏与悬空指针问题​ 4.3 new和delete的重载​ 四、智能指针…...

vulnyx Blogger writeup

信息收集 arp-scan nmap 获取userFlag 上web看看 一个默认的页面,gobuster扫一下目录 可以看到扫出的目录中得到了一个有价值的目录/wordpress,说明目标所使用的cms是wordpress,访问http://192.168.43.213/wordpress/然后查看源码能看到 这…...

Selenium常用函数介绍

目录 一,元素定位 1.1 cssSeector 1.2 xpath 二,操作测试对象 三,窗口 3.1 案例 3.2 窗口切换 3.3 窗口大小 3.4 屏幕截图 3.5 关闭窗口 四,弹窗 五,等待 六,导航 七,文件上传 …...

2025年渗透测试面试题总结-腾讯[实习]科恩实验室-安全工程师(题目+回答)

安全领域各种资源,学习文档,以及工具分享、前沿信息分享、POC、EXP分享。不定期分享各种好玩的项目及好用的工具,欢迎关注。 目录 腾讯[实习]科恩实验室-安全工程师 一、网络与协议 1. TCP三次握手 2. SYN扫描原理 3. HTTPS证书机制 二…...

【Linux系统】Linux环境变量:系统配置的隐形指挥官

。# Linux系列 文章目录 前言一、环境变量的概念二、常见的环境变量三、环境变量特点及其相关指令3.1 环境变量的全局性3.2、环境变量的生命周期 四、环境变量的组织方式五、C语言对环境变量的操作5.1 设置环境变量:setenv5.2 删除环境变量:unsetenv5.3 遍历所有环境…...