C++学习之路(六)C++ 实现简单的工具箱系统命令行应用 - 示例代码拆分讲解

简单的工具箱系统示例介绍:

这个示例展示了一个简单的工具箱框架,它涉及了几个关键概念和知识点:

  1. 面向对象编程 (OOP):使用了类和继承的概念。Tool 是一个纯虚类,CalculatorToolFileReaderTool 是其派生类。

  2. 多态:通过将不同类型的工具对象放入同一个容器中,实现了多态,能够根据用户输入执行不同的工具。

  3. STL 容器 (std::map):使用了 std::map 作为工具箱的存储容器,将工具名称映射到相应的工具对象。

  4. 文件输入输出:在 FileReaderTool 中展示了如何使用文件输入输出流来读取文件内容。

  5. 异常处理:在 CalculatorTool 中检查除数是否为零,避免出现除以零的情况。

这个示例提供了一个基础框架,能够让用户扩展和添加新的工具到工具箱中,并在运行时选择和执行这些工具。它展示了如何使用面向对象的思想创建可扩展的程序结构,并利用 C++ 的特性和标准库来实现这一点。
在这里插入图片描述

功能和概述:

  1. 工具箱功能:用户可以通过添加不同的工具(例如计算器、文件读取器)扩展工具箱。每个工具都执行特定的任务。

  2. 用户交互:程序在运行时会列出所有可用的工具,并要求用户输入要执行的工具名称。用户可以通过输入工具名称来选择要执行的功能。

  3. 多态执行:使用多态性,不同类型的工具对象存储在同一个容器中,通过执行基类 Tool 的虚函数 execute() 来实现不同工具的执行。

  4. 异常处理:在计算器工具中,程序会检查用户输入的除数是否为零,并进行相应的异常处理,避免出现除以零的情况。

  5. C++ 特性应用:使用了面向对象编程的概念,包括类、继承和虚函数,以及标准库中的 std::map 和输入输出流。

这个框架提供了一个基础结构,可用作添加更多功能的起点,用户可以根据需求扩展和修改工具箱,并通过添加新的工具类来实现更多的功能。


示例在Clion中运行步骤:

1. 新建项目

在这里插入图片描述

2. 粘贴代码
#include <iostream>
#include <map>
#include <functional>
#include <fstream>// 工具基类
class Tool {
public:virtual void execute() = 0;virtual ~Tool() = default;
};// 示例工具类:计算器
class CalculatorTool : public Tool {
public:void execute() override {std::cout << "Executing Calculator Tool..." << std::endl;double num1, num2;char op;std::cout << "Enter first number: ";std::cin >> num1;std::cout << "Enter operator (+, -, *, /): ";std::cin >> op;std::cout << "Enter second number: ";std::cin >> num2;double result;switch (op) {case '+':result = num1 + num2;break;case '-':result = num1 - num2;break;case '*':result = num1 * num2;break;case '/':if (num2 != 0) {result = num1 / num2;} else {std::cout << "Error! Division by zero is not allowed." << std::endl;return;}break;default:std::cout << "Invalid operator." << std::endl;return;}std::cout << "Result: " << result << std::endl;}
};// 示例工具类:文件读取器
class FileReaderTool : public Tool {
public:void execute() override {std::cout << "Executing File Reader Tool..." << std::endl;std::string filename;std::cout << "Enter file name to read: ";std::cin >> filename;std::ifstream file(filename);if (file.is_open()) {std::string line;while (std::getline(file, line)) {std::cout << line << std::endl;}file.close();} else {std::cout << "Failed to open file." << std::endl;}}
};// 工具箱
class Toolbox {
private:std::map<std::string, std::unique_ptr<Tool>> tools;public:// 添加工具到工具箱void addTool(const std::string &name, std::unique_ptr<Tool> tool) {tools[name] = std::move(tool);}// 列出所有可用工具void listTools() {std::cout << "Available tools: ";for (const auto &tool : tools) {std::cout << tool.first << ", ";}std::cout << std::endl;}// 执行特定工具void executeTool(const std::string &name) {auto it = tools.find(name);if (it != tools.end()) {it->second->execute();} else {std::cout << "Tool not found." << std::endl;}}
};int main() {Toolbox toolbox;// 创建工具对象并添加到工具箱中toolbox.addTool("Calculator", std::make_unique<CalculatorTool>());toolbox.addTool("FileReader", std::make_unique<FileReaderTool>());// 列出所有可用工具toolbox.listTools();// 执行特定工具std::string selectedTool;std::cout << "Enter tool name to execute: ";std::cin >> selectedTool;toolbox.executeTool(selectedTool);return 0;
}
3. 编译运行

在这里插入图片描述
在这里插入图片描述

代码拆解,知识点总结

让我们逐步分解和解释这个示例:

🟥 1. 引入头文件和命名空间

#include <iostream>
#include <map>
#include <functional>
#include <fstream>

这部分代码引入了所需的标准库头文件,包括用于输入输出的 iostream,用于映射的 map,用于函数对象的 functional 和文件流的 fstream。同时,使用了 std 命名空间。


Tips: 📢 什么是 functional ?

<functional> 是 C++ 标准库中的头文件,它提供了一组函数对象,以及用于操作函数的工具。函数对象是可调用对象(callable object),可以像函数一样使用。这个头文件提供了很多功能,主要包括以下几个方面:

  1. 函数对象 (Function Objects):包括各种函数对象,如 std::functionstd::bindstd::placeholders 等。这些函数对象允许您以更灵活的方式处理函数,例如将函数作为参数传递给其他函数,或者将函数与特定参数绑定在一起。

  2. 函数适配器 (Function Adapters):提供了一些函数适配器,例如 std::bindstd::mem_fn,它们允许您更改函数的行为或结构。

  3. 一些工具性质的功能:比如 std::reference_wrapper,它允许将引用封装为可复制的对象,方便函数接受引用参数。

使用 <functional> 头文件,您可以更方便地处理函数式编程、函数组合和函数对象。例如,std::function 允许您将函数作为参数传递给其他函数,std::bind 允许您绑定参数到函数,创建新的函数对象等。

这个头文件提供了许多功能,使得 C++ 中的函数操作更加灵活和便捷。


🟥 2. Tool

class Tool {
public:virtual void execute() = 0;virtual ~Tool() = default;
};

这里定义了一个抽象基类 Tool,它包含一个纯虚函数 execute(),这个函数没有实现,需要在派生类中具体实现。另外,声明了虚析构函数,使得这个类可以作为多态基类。


Tips: 📢 virtual void 和 virtual ~Tool() = default 是什么意思?

这两个内容涉及到 C++ 中的虚函数和虚析构函数的概念。

virtual void execute() = 0;

这行代码定义了一个纯虚函数 execute()。虚函数是在基类中声明为虚函数的函数,在派生类中可以进行重写。纯虚函数没有具体的实现,它的目的是让派生类强制实现该函数。类中包含纯虚函数的类称为抽象类,抽象类不能被实例化,只能被用作其他类的基类。在这个例子中,Tool 类定义了一个纯虚函数 execute(),所有继承自 Tool 的类都必须提供自己的 execute() 函数的实现。

virtual ~Tool() = default;

这行代码定义了一个虚析构函数。虚析构函数是用来释放内存的,通常当一个基类指针指向一个派生类对象时,如果基类的析构函数不是虚函数,那么当通过基类指针删除对象时,只会调用基类的析构函数而不会调用派生类的析构函数,可能会导致内存泄漏。所以将基类的析构函数声明为虚函数能够确保通过基类指针删除派生类对象时正确地调用派生类的析构函数。virtual ~Tool() = default; 表示使用默认实现的虚析构函数,即使用编译器生成的默认析构函数。

为什么要写成 virtual void execute() = 0 ?

在 C++ 中,将一个函数声明为纯虚函数(Pure Virtual Function)可以通过在函数声明末尾加上 = 0 来实现。这种写法告诉编译器,这个函数在基类中没有实际的实现,派生类必须提供自己的实现才能创建对象。

纯虚函数实际上是一个占位符,它为派生类提供了一个契约:如果你要成为这个基类的子类,你必须提供这个函数的实际实现。这种机制使得基类能够定义一组接口,但不提供具体的实现,而是将实现的责任交给派生类。

写成 = 0 的语法是 C++ 中定义纯虚函数的标准方式。这样做有两个主要原因:

  1. 为了让编译器理解这是一个纯虚函数,因此任何派生类都需要提供它的实现。
  2. 如果一个类包含了纯虚函数,它就不能被直接实例化,只能被用作其他类的基类,这种情况下它就成为了抽象类。

因此,使用 = 0 是 C++ 中约定的方法,告诉编译器这是一个纯虚函数。


🟥 3. 示例工具类:CalculatorToolFileReaderTool

class CalculatorTool : public Tool {
public:void execute() override {// ... 实现了简单的计算器功能 ...}
};class FileReaderTool : public Tool {
public:void execute() override {// ... 实现了文件读取器功能 ...}
};

这部分代码展示了两个派生类,分别是 CalculatorToolFileReaderTool。它们继承自 Tool 类,并重写了 execute() 方法,实现了各自的功能。

Tips: 📢 加 public 是什么意思?还有什么其他方式吗?

这段代码定义了一个名为 CalculatorTool 的类,它是 Tool 类的公有派生类。这意味着 CalculatorTool 类继承了 Tool 类的成员和方法,并且可以在其基础上添加自己的成员和方法。

这种继承关系使得 CalculatorTool 类拥有 Tool 类的特性,并且可以通过重写(覆盖)Tool 类中的虚函数来实现自己特定的行为。继承允许在不重复编写相同代码的情况下,扩展已有类的功能,这有助于代码的复用和扩展。

在这里,CalculatorTool 类继承了 Tool 类的一些特性,可能包括一些成员函数、虚函数或其他成员。同时,CalculatorTool 类可以添加自己特有的功能,例如实现 execute() 函数以执行特定的计算器功能。


在 C++ 中,public 是一种访问控制修饰符,它用于指定类成员的访问级别。在这里,class CalculatorTool : public Tool 中的 public 关键字表示派生类 CalculatorTool 继承自基类 Tool 并且继承的是基类中的公有成员。

访问控制修饰符包括三种:publicprotectedprivate。它们决定了派生类对基类的继承成员的访问权限:

  • public 继承意味着基类中的公有成员在派生类中仍然是公有的,基类的保护和私有成员在派生类中不可直接访问。
  • protected 继承意味着基类中的公有和保护成员在派生类中都变为保护成员,基类的私有成员在派生类中不可直接访问。
  • private 继承意味着基类中的所有成员都变为派生类的私有成员,不可在派生类之外访问。

除了 public 继承之外,还有 protectedprivate 继承。这些继承方式决定了派生类对基类成员的访问权限。

class Derived : protected Base {// 在这里,Derived 是 protected 继承自 Base// Base 中的公有成员在 Derived 中变成了 protected
};class Derived : private Base {// 在这里,Derived 是 private 继承自 Base// Base 中的所有成员在 Derived 中都变成了 private
};

但一般情况下,使用 public 继承是最常见和推荐的方式,因为它保持了基类的接口,并且能够更加清晰地表达派生类和基类之间的关系。


🟥 4. Toolbox

class Toolbox {
private:std::map<std::string, std::unique_ptr<Tool>> tools;public:void addTool(const std::string &name, std::unique_ptr<Tool> tool) {// ... 将工具名称和对象关联存储到map中 ...}void listTools() {// ... 列出所有可用工具 ...}void executeTool(const std::string &name) {// ... 执行特定工具 ...}
};

这个类用来管理工具。它包含一个私有成员 tools,这是一个 map,将工具名称与对应的工具对象关联起来。addTool() 方法用于向工具箱中添加工具,listTools() 方法用于列出所有可用的工具,executeTool() 方法根据用户输入的名称执行相应的工具。


Tips: 📢 有关 std::map 知识点讲解

std::map<std::string, std::unique_ptr<Tool>> tools; 这段代码定义了一个名为 tools 的成员变量,它是一个 std::map 对象。让我来解释一下:

  • std::map 是 C++ 标准库中的关联容器,它提供了键值对(key-value)的存储机制,允许使用一个键来快速检索相关联的值。
  • 在这个例子中,std::map 使用了两个模板参数:std::stringstd::unique_ptr<Tool>。它将字符串(std::string)作为键,将指向 Tool 类对象的独占指针(std::unique_ptr<Tool>)作为值。
  • 意思是 tools 这个 map 变量能够通过字符串键来快速存储和检索不同的工具对象。

例如,可以用这个 tools map 来存储不同的工具,每个工具都有一个与之对应的字符串名称。这样,通过工具的名字就能够方便地获取对应的工具对象。

例如:

std::map<std::string, std::unique_ptr<Tool>> tools;// 添加一个名为 "calculator" 的计算器工具
tools["calculator"] = std::make_unique<CalculatorTool>();// 添加一个名为 "fileReader" 的文件读取器工具
tools["fileReader"] = std::make_unique<FileReaderTool>();// 通过名称获取特定工具
std::string selectedToolName = "calculator";
auto selectedTool = tools[selectedToolName]; // 获取名为 "calculator" 的工具
selectedTool->execute(); // 执行对应的工具操作

这段代码展示了如何向 tools map 中添加不同的工具,并通过名称检索和执行特定的工具操作。


🟥 5. main() 函数

int main() {Toolbox toolbox;// ... 添加工具到工具箱中 ...toolbox.listTools();std::string selectedTool;std::cout << "Enter tool name to execute: ";std::cin >> selectedTool;toolbox.executeTool(selectedTool);return 0;
}

main() 函数中,创建了一个 Toolbox 对象,并添加了具体的工具到工具箱中。然后列出了所有可用的工具,等待用户输入工具名称,最后执行用户选择的工具。

这个拆解旨在展示这个程序的基本结构和主要组成部分。每个部分都有其特定的功能,通过彼此协作来实现整体的功能。


Tips: 📢 别忘了跑起来,检查检查有没有BUG ~ 😁


本文就到这里了,感谢您的阅读,明天还有更多的实例学习文章等着你 🎆。别忘了点赞、收藏~ Thanks♪(・ω・)ノ 🍇。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/173078.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

csdn的搜索关键词

忽然发现一个问题&#xff1a;通过关键字找自己的博文&#xff0c;自己知道自己曾经写过某篇文章&#xff0c;但是不记得具体目录与位置&#xff0c;标题了&#xff0c;只知道大概是什么相关的&#xff0c;这个时候就需要关键字查询了&#xff0c;快速方便的找到自己需要的文章…

计算一个Series序列的n阶滞后相关系数Series.autocorr()

【小白从小学Python、C、Java】 【计算机等考500强证书考研】 【Python-数据分析】 计算一个时间序列的 n阶滞后自相关系数 Series.autocorr(n) [太阳]选择题 以下代码的说法中正确的是? import pandas as pd s1 pd.Series([1,2,3,4,5,6]) print("【显示】s1:\n",…

7 通用数字量输入输出GPIO

文章目录 7.0 GPIO概念7.1 GPIO工作原理7.2 GPIO寄存器以及编程7.2.5 GPIO寄存器编程设置与应用 7.3 GPIO跑马灯7.3.1 LED 输出初始化7.3.2 跑马灯输出实验7.3.3 按键输入实验 7.0 GPIO概念 GPIO&#xff08;general purpose intput output&#xff09;是通用输入输出端口的简…

最火web大屏可视化编辑器

前言&#xff1a; 乐吾乐Le5le大屏可视化设计器&#xff0c;零代码实现物联网、工业智能制造等领域的可视化大屏、触摸屏端UI以及工控可视化的解决方案。同时也是一个Web组态工具&#xff0c;支持2D、3D等多种形式&#xff0c;用于构建具有实时数据展示、监控预警、丰富交互的组…

2009年iMac装64位windows7及win10

2009年iMac装64位windows7及win10 Boot Camp没有“创建 Windows7 或更高版本的安装磁盘”选项 安装完Mac OS系统后&#xff0c;要制作Windows7安装U盘时才发现&#xff0c;Boot Camp没有“创建 Windows7 或更高版本的安装磁盘”选项&#xff0c;搜索到文章&#xff1a;修改Boo…

laravel框架学习

一、文件上传 在控制器中按下面所示书写 public function upload(){$this->domain $_SERVER[HTTP_HOST]; //获取当前域我&#xff0c; 其实这个是不应该写在这儿&#xff08;应该是一个全局&#xff09;&#xff0c;我只是做个例子。$file $this->require->file(fi…

blender 3D眼球结构

角膜&#xff08;Cornea&#xff09;&#xff1a;眼球的前部&#xff0c;透明的曲面&#xff0c;负责折射光线。虹膜&#xff08;Iris&#xff09;&#xff1a;眼睛的颜色部分&#xff0c;控制瞳孔大小以调整进入眼睛的光量。瞳孔&#xff08;Pupil&#xff09;&#xff1a;虹膜…

Mycat实现读写分离

Mycat实现读写分离 Mycat支持MySQL主从复制状态绑定的读写分离机制。这里实现的也是基于MySQL主从复制的读写分离。 MySQL主从复制配置 首先要配置MySQL的主从复制&#xff0c;这里配置的是一主一次从。可以参考下面的文章。 https://blog.csdn.net/wsb_2526/article/detail…

Zookeeper分布式锁实现Curator十一问

前面我们通过Redis分布式锁实现Redisson 15问文章剖析了Redisson的源码&#xff0c;理清了Redisson是如何实现的分布式锁和一些其它的特性。这篇文章就来接着剖析Zookeeper分布式锁的实现框架Curator的源码&#xff0c;看看Curator是如何实现Zookeeper分布式锁的&#xff0c;以…

OpenMMlab导出yolox模型并用onnxruntime和tensorrt推理

导出onnx文件 直接使用脚本 import torch from mmdet.apis import init_detector, inference_detectorconfig_file ./configs/yolox/yolox_tiny_8xb8-300e_coco.py checkpoint_file yolox_tiny_8x8_300e_coco_20211124_171234-b4047906.pth model init_detector(config_fi…

ARM - AArch64 - 通用寄存器

说明 在深入一点了解了系统调用以及非安全world&#xff08;REE&#xff09;/安全world&#xff08;TEE&#xff09;切换时参数传递和结果返回的实现原理&#xff08;通过通用寄存器实现&#xff09;&#xff0c;对通用寄存器的使用有了一个全新的认识&#xff0c;对知识做个总…

MATLAB中corrcoef函数用法

目录 语法 说明 示例 矩阵的随机列 两个随机变量 矩阵的 P 值 相关性边界 NaN 值 corrcoef函数的功能是返回数据的相关系数。 语法 R corrcoef(A) R corrcoef(A,B) [R,P] corrcoef(___) [R,P,RL,RU] corrcoef(___) ___ corrcoef(___,Name,Value) 说明 R corrc…

【海德教育】唐山成人高考艺术类包括哪些专业?

成人高校开设的艺术类招生专业主要有&#xff1a;艺术设计、装饰、装潢、书法、绘画、音乐、美术、戏剧表演、播音、服装设计、摄影等专业。

sebp/elk镜像历史版本

最近因为之前sebp/elk的镜像和容器出现问题而误删了. 新版本随便功能全, 但是配置较为繁琐. 因此想要根据之前的截图找到之前的版本. 但是查看版本需要科学. 因此在花时间研究出来科学的方法之后, 还是决定将历史的版本信息留存下来, 以供后续开发需要 相关官网(需科学上网): s…

1. 图的广度优先遍历

当一道题的AC变成了找不同的时候&#xff0c;一切就开始失去意义。 到底是谁&#xff1f;把Search写成Seacrh&#xff0c;害我一直找不同。 本实验实现邻接表表示下无向图的广度优先遍历。 程序的输入是图的顶点序列和边序列(顶点序列以*为结束标志&#xff0c;边序列以-1,-1…

【洛谷算法题】P5715-三位数排序【入门2分支结构】

&#x1f468;‍&#x1f4bb;博客主页&#xff1a;花无缺 欢迎 点赞&#x1f44d; 收藏⭐ 留言&#x1f4dd; 加关注✅! 本文由 花无缺 原创 收录于专栏 【洛谷算法题】 文章目录 【洛谷算法题】P5715-三位数排序【入门2分支结构】&#x1f30f;题目描述&#x1f30f;输入格式…

代码随想录-刷题第九天

28. 找出字符串中第一个匹配项的下标 题目链接&#xff1a;28. 找出字符串中第一个匹配项的下标 思路1&#xff1a;先来写一下暴力解法。 时间复杂度O(n*m) class Solution {public int strStr(String haystack, String needle) {// 暴力解法先来一遍for (int i 0; i <…

【Seata源码学习 】篇五 注册分支事务

【Seata源码学习 】篇五 分支事务注册 1.远程服务调用绑定XID 回到事务模版方法类TransactionalTemplate中 beginTransaction(txInfo, tx);Object rs;try {// Do Your Business// 执行执行拦截器链路rs business.execute();} catch (Throwable ex) {// 3. The needed busine…

基于数据挖掘的智能停车场运营数据分析系统(毕业论文)

点击完整下载 基于数据挖掘的智能停车场运营数据分析系统 "A Data Mining-Based Intelligent Parking Lot Operational Data Analysis System" 目录 目录 2 摘要 3 关键词 4 第一章 绪论 4 1.1 研究背景 4 1.2 研究意义 5 1.3 主要研究内容 7 1.4 研究方法与流程 8 1…

基于Java SSM框架+Vue留学生交流互动论坛网站项目【项目源码+论文说明】计算机毕业设计

基于java的SSM框架Vue实现学生交流互动论坛网站演示 摘要 21世纪的今天&#xff0c;随着社会的不断发展与进步&#xff0c;人们对于信息科学化的认识&#xff0c;已由低层次向高层次发展&#xff0c;由原来的感性认识向理性认识提高&#xff0c;管理工作的重要性已逐渐被人们所…