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

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♪(・ω・)ノ 🍇。

相关文章:

  • KVM虚拟机的NAT网络模式原理及过程展示
  • Android frameworks 开发总结之九(Settings)
  • MySQL与Redis如何保证数据的一致性
  • rust tokio select!宏详解
  • python爬虫进阶篇(异步)
  • 第十七章 处理空字符串和 Null 值 - XMLIGNORENULL、XMLNIL 和 XMLUSEMPTYELEMENT 的详细信息
  • C#,《小白学程序》第二十一课:大数的减法(BigInteger Subtract)
  • 第十六章 解读深度学习中Batch Size、Iterations和Epochs(工具)
  • 博物馆线上导览系统的设计与实现-计算机毕业设计源码64574
  • 【管理运筹学】背诵手册(六)| 图与网络分析(基本概念、最小支撑树问题、最短路问题)
  • 播放器开发(三):FFmpeg与SDL环境配置
  • 基于C#实现双端队列
  • Centos开机启动Java程序
  • Drool 7 SpreadSheet Decision Template 笔记
  • Jenkins用126邮箱发邮件为什么发不出去
  • Angularjs之国际化
  • Java-详解HashMap
  • LeetCode541. Reverse String II -- 按步长反转字符串
  • Linux CTF 逆向入门
  • Lucene解析 - 基本概念
  • orm2 中文文档 3.1 模型属性
  • Redis 懒删除(lazy free)简史
  • Redis 中的布隆过滤器
  • seaborn 安装成功 + ImportError: DLL load failed: 找不到指定的模块 问题解决
  • vuex 学习笔记 01
  • 函数式编程与面向对象编程[4]:Scala的类型关联Type Alias
  • 讲清楚之javascript作用域
  • 力扣(LeetCode)22
  • 使用Swoole加速Laravel(正式环境中)
  • 一些基于React、Vue、Node.js、MongoDB技术栈的实践项目
  • 原生JS动态加载JS、CSS文件及代码脚本
  • 基于django的视频点播网站开发-step3-注册登录功能 ...
  • 智能情侣枕Pillow Talk,倾听彼此的心跳
  • ###C语言程序设计-----C语言学习(6)#
  • #Lua:Lua调用C++生成的DLL库
  • #我与Java虚拟机的故事#连载11: JVM学习之路
  • $HTTP_POST_VARS['']和$_POST['']的区别
  • ( 10 )MySQL中的外键
  • (1)Android开发优化---------UI优化
  • (C#)Windows Shell 外壳编程系列4 - 上下文菜单(iContextMenu)(二)嵌入菜单和执行命令...
  • (二)linux使用docker容器运行mysql
  • (二)正点原子I.MX6ULL u-boot移植
  • (黑马出品_高级篇_01)SpringCloud+RabbitMQ+Docker+Redis+搜索+分布式
  • (离散数学)逻辑连接词
  • (算法)Game
  • (原創) 如何安裝Linux版本的Quartus II? (SOC) (Quartus II) (Linux) (RedHat) (VirtualBox)
  • (转)自己动手搭建Nginx+memcache+xdebug+php运行环境绿色版 For windows版
  • (转载)OpenStack Hacker养成指南
  • .[backups@airmail.cc].faust勒索病毒的最新威胁:如何恢复您的数据?
  • .NET 5种线程安全集合
  • .NET Core使用NPOI导出复杂,美观的Excel详解
  • .NET 反射 Reflect
  • .NET6 命令行启动及发布单个Exe文件
  • .NET高级面试指南专题十一【 设计模式介绍,为什么要用设计模式】
  • ??在JSP中,java和JavaScript如何交互?