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. 粘贴代码
cpp 复制代码
#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. 引入头文件和命名空间

cpp 复制代码
#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

cpp 复制代码
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

cpp 复制代码
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 继承。这些继承方式决定了派生类对基类成员的访问权限。

cpp 复制代码
class Derived : protected Base {
    // 在这里,Derived 是 protected 继承自 Base
    // Base 中的公有成员在 Derived 中变成了 protected
};

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

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


🟥 4. Toolbox

cpp 复制代码
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 来存储不同的工具,每个工具都有一个与之对应的字符串名称。这样,通过工具的名字就能够方便地获取对应的工具对象。

例如:

cpp 复制代码
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() 函数

cpp 复制代码
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♪(・ω・)ノ 🍇。

相关推荐
虾球xz1 分钟前
游戏引擎学习第58天
学习·游戏引擎
CodeClimb11 分钟前
【华为OD-E卷-木板 100分(python、java、c++、js、c)】
java·javascript·c++·python·华为od
CT随22 分钟前
Redis内存碎片详解
java·开发语言
anlog31 分钟前
C#在自定义事件里传递数据
开发语言·c#·自定义事件
奶香臭豆腐44 分钟前
C++ —— 模板类具体化
开发语言·c++·学习
不想当程序猿_1 小时前
【蓝桥杯每日一题】分糖果——DFS
c++·算法·蓝桥杯·深度优先
晚夜微雨问海棠呀1 小时前
长沙景区数据分析项目实现
开发语言·python·信息可视化
graceyun1 小时前
C语言初阶习题【9】数9的个数
c语言·开发语言
cdut_suye1 小时前
Linux工具使用指南:从apt管理、gcc编译到makefile构建与gdb调试
java·linux·运维·服务器·c++·人工智能·python
波音彬要多做1 小时前
41 stack类与queue类
开发语言·数据结构·c++·学习·算法