[C++]C++工具之对异常情况的处理(throw、catch、try)以及用命名空间避免同名冲突

一、C++ 异常处理😊


1.1 定义

C++ 中的异常处理用于应对程序运行中的异常情况(如除零、数组越界等),通过 try-catch 机制捕获和处理错误,防止程序崩溃。
异常是程序运行时意外发生的事件,可以通过抛出(throw)和捕获(catch)机制处理。


1.2 通俗解释

  • 抛异常(throw): 就像在程序运行时发现问题了,扔出一个问题给系统处理。
  • 捕获异常(catch): 系统接收到这个问题后,根据你的代码去解决问题。
  • try 块: 这是一个保护块,把可能出错的代码放在这里,避免程序直接崩溃。

1.3 固定格式

try {// 可能抛出异常的代码throw exception_object;  // 抛出异常
} catch (exception_type variable_name) {// 捕获异常并处理
} catch (...) {// 捕获所有类型的异常(可选)
}
  • throw:用来抛出异常,可以是内置类型(如整数、字符串)或自定义类型(如对象)。
  • catch:用来捕获异常,必须匹配 throw 的类型,或者用 ... 捕获所有类型。
  • try:将可能出错的代码块包裹起来。

1.4 注意点

  1. 匹配顺序:

    • catch 子句是按顺序匹配的,第一个匹配的会被执行。
    • catch (...) 必须放在最后,用于捕获所有未明确处理的异常。
  2. 异常传递:

    • 如果函数中抛出的异常没有被处理,会向调用者函数传播,直到找到合适的 catch 块。
  3. 自定义异常类:

    • 可以通过定义自己的类,抛出更具体的异常信息。

1.5C++中的主要异常类型

1.5.1. 内置类型的异常

1. 整数异常

可以直接抛出整数类型的异常,用于简单错误标识。

#include <iostream>
using namespace std;int main() {try {throw 42;  // 抛出整数异常} catch (int e) {  // 捕获整数类型的异常cout << "Caught integer exception: " << e << endl;}return 0;
}
/*
输出:
Caught integer exception: 42
*/

2. 字符串异常

可以抛出字符串(如 const char*std::string)类型的异常,通常用来描述错误信息。

#include <iostream>
using namespace std;int main() {try {throw "An error occurred";  // 抛出字符串异常} catch (const char* e) {  // 捕获字符串类型异常cout << "Caught exception: " << e << endl;}return 0;
}
/*
输出:
Caught exception: An error occurred
*/

3. 浮点数异常

可以直接抛出浮点类型,用于表示数值相关的错误。

#include <iostream>
using namespace std;int main() {try {throw 3.14;  // 抛出浮点数异常} catch (double e) {  // 捕获浮点类型异常cout << "Caught floating-point exception: " << e << endl;}return 0;
}
/*
输出:
Caught floating-point exception: 3.14
*/

4. 布尔类型异常

布尔类型也可以被抛出,通常用作简单的状态标识。

#include <iostream>
using namespace std;int main() {try {throw true;  // 抛出布尔类型异常} catch (bool e) {cout << "Caught boolean exception: " << (e ? "true" : "false") << endl;}return 0;
}
/*
输出:
Caught boolean exception: true
*/


1.5.2. 标准库类型的异常

C++ 提供了一些预定义的异常类,它们位于标准库中,包含在 <stdexcept><exception> 等头文件中。以下是常用的标准异常类型:

1. std::exception
  • 是所有标准异常类的基类。
  • 通常作为通用异常捕获的类型。
    #include <iostream>
    #include <exception>
    using namespace std;int main() {try {throw exception();  // 抛出标准异常} catch (exception& e) {  // 捕获标准异常cout << "Caught standard exception: " << e.what() << endl;}return 0;
    }
    /*
    输出:
    Caught standard exception: std::exception
    */

    2. std::logic_error
  • 表示程序逻辑错误,通常由于程序员的疏忽引起。
  • 例如:访问空指针、非法参数等。
#include <iostream>
#include <stdexcept>
using namespace std;int main() {try {throw logic_error("Logic error occurred");  // 抛出逻辑错误} catch (logic_error& e) {cout << "Caught logic_error: " << e.what() << endl;}return 0;
}
/*
输出:
Caught logic_error: Logic error occurred
*/
 3. std::runtime_error
  • 表示程序运行时的错误。
  • 例如:文件未找到、网络中断等。
#include <iostream>
#include <stdexcept>
using namespace std;int main() {try {throw runtime_error("Runtime error occurred");  // 抛出运行时错误} catch (runtime_error& e) {cout << "Caught runtime_error: " << e.what() << endl;}return 0;
}
/*
输出:
Caught runtime_error: Runtime error occurred
*/

4. std::bad_alloc
  • 表示内存分配失败。
  • new 操作符无法分配内存时,抛出此异常。
#include <iostream>
#include <new>
using namespace std;int main() {try {int* arr = new int[1000000000000000];  // 请求过多内存} catch (bad_alloc& e) {cout << "Caught bad_alloc: " << e.what() << endl;}return 0;
}
/*输出:
Caught bad_alloc: std::bad_alloc
*/

5. std::out_of_range
  • 表示访问容器时超出有效范围。
  • 例如:访问数组中不存在的元素。
#include <iostream>
#include <vector>
#include <stdexcept>
using namespace std;int main() {try {vector<int> vec = {1, 2, 3};cout << vec.at(10);  // 超出范围} catch (out_of_range& e) {cout << "Caught out_of_range: " << e.what() << endl;}return 0;
}
/*输出:
Caught out_of_range: vector::_M_range_check: __n (which is 10) >= this->size() (which is 3)
*/

1.5.3自定义类型的异常

用户可以自定义类作为异常类型,用于处理程序特定的错误场景。

1.自定义异常类
#include <iostream>
#include <string>
using namespace std;class MyException {string message;
public:MyException(string msg) : message(msg) {}string getMessage() const { return message; }
};int main() {try {throw MyException("Custom exception occurred!");  // 抛出自定义异常} catch (MyException& e) {cout << "Caught exception: " << e.getMessage() << endl;}return 0;
}
/*输出:
Caught exception: Custom exception occurred!
*/

1.5.4 捕获所有类型的异常

1.使用 catch(...) 捕获任意类型的异常。
#include <iostream>
using namespace std;int main() {try {throw 3.14;  // 抛出浮点数异常} catch (...) {  // 捕获所有异常cout << "Caught an unknown exception." << endl;}return 0;
}
/*输出:
Caught an unknown exception.
*/

1.6总结

常见的异常类型

1.6.1内置类型:

  1. 整数、浮点数、字符串、布尔值。

1.6.2标准库类型:

  1. std::exception(基类)。
  2. std::logic_error(逻辑错误)。
  3. std::runtime_error(运行时错误)。
  4. std::bad_alloc(内存分配失败)。
  5. std::out_of_range(超出范围错误)。

1.6.3用户自定义类型:

  1. 用户可以定义自己的类来表示异常类型。

1.7捕获规则

  1. 使用匹配的 catch 处理具体异常。
  2. 使用 catch(...) 捕获所有未处理的异常。

1.7.1综合代码示例与解析:

#include <iostream>
#include <string>
#include <vector>
#include <stdexcept>
using namespace std;// 自定义异常类
class MyException {string message;
public:MyException(string msg) : message(msg) {}string getMessage() const { return message; }
};int main() {try {// Uncomment (取消注释) 以下一行代码逐个测试不同类型的异常// throw 42;                          // 整数类型异常// throw 3.14;                        // 浮点数类型异常// throw "An error occurred!";        // 字符串异常// throw string("String object error"); // std::string 异常// throw logic_error("Logic error");  // 标准库逻辑错误// throw runtime_error("Runtime error"); // 标准库运行时错误// throw out_of_range("Out of range error"); // 超出范围异常// throw bad_alloc();                 // 内存分配失败异常throw MyException("Custom exception occurred!"); // 自定义异常} catch (int e) {cout << "Caught an integer exception: " << e << endl;} catch (double e) {cout << "Caught a floating-point exception: " << e << endl;} catch (const char* e) {cout << "Caught a C-string exception: " << e << endl;} catch (string& e) {cout << "Caught a string object exception: " << e << endl;} catch (logic_error& e) {cout << "Caught logic_error: " << e.what() << endl;} catch (runtime_error& e) {cout << "Caught runtime_error: " << e.what() << endl;} catch (out_of_range& e) {cout << "Caught out_of_range: " << e.what() << endl;} catch (bad_alloc& e) {cout << "Caught bad_alloc: " << e.what() << endl;} catch (MyException& e) {cout << "Caught custom exception: " << e.getMessage() << endl;} catch (...) { // 捕获所有其他类型的异常cout << "Caught an unknown exception." << endl;}cout << "Program continues after exception handling." << endl;return 0;
}

1.7.2运行结果示例:

1. 抛出整数类型异常:throw 42;

输出:

Caught an integer exception: 42
Program continues after exception handling.

2. 抛出浮点数类型异常:throw 3.14;

输出:

Caught a floating-point exception: 3.14
Program continues after exception handling.

3. 抛出 C 字符串类型异常:throw "An error occurred!";

输出:

Caught a C-string exception: An error occurred!
Program continues after exception handling.

4. 抛出 C++ 字符串类型异常:throw string("String object error");

输出:

Caught a string object exception: String object error
Program continues after exception handling.
5. 抛出标准库逻辑错误异常:throw logic_error("Logic error");

输出:

Caught logic_error: Logic error
Program continues after exception handling.

6. 抛出标准库运行时错误异常:throw runtime_error("Runtime error");

输出:

Caught runtime_error: Runtime error
Program continues after exception handling.

7. 抛出超出范围异常:throw out_of_range("Out of range error");

输出:

Caught out_of_range: Out of range error
Program continues after exception handling.

8. 抛出内存分配失败异常:throw bad_alloc();

输出:

Caught bad_alloc: std::bad_alloc
Program continues after exception handling.

9. 抛出自定义异常:throw MyException("Custom exception occurred!");

输出:

Caught custom exception: Custom exception occurred!
Program continues after exception handling.

10. 抛出未匹配的异常(如结构体或未列出的类型);
struct UnknownException {};
throw UnknownException();

输出:

Caught an unknown exception.
Program continues after exception handling.

1.7.3通用总结

  1. 对每种异常类型都有专门的 catch 块处理,并输出对应的内容。
  2. 如果异常类型不匹配,会进入通用的 catch(...) 块,确保程序不会因未捕获的异常而崩溃。

解析: 

  1. 逐个抛出不同类型的异常:

    1. 使用 throw 语句抛出整数、浮点数、字符串、标准库异常、自定义异常等。
  2. 多个 catch 块:

    1. 每个 catch 块捕获一种特定类型的异常。
    2. 使用 catch(...) 捕获所有未明确处理的异常。
  3. 自定义异常:

    1. 定义了一个 MyException 类,用于抛出和捕获自定义的错误。

二、用命名空间避免同名冲突


1. 什么是同名冲突?

1.1 同名冲突现象

在大型项目中,不同的模块或库可能会定义相同名字的变量、函数或类。这会导致程序不知道该调用哪一个。例如:

#include <iostream>
using namespace std;void print() {cout << "Global print function" << endl;
}namespace ModuleA {void print() {cout << "ModuleA's print function" << endl;}
}int main() {print();  // 问题:调用的是全局的 print 还是 ModuleA 的 print?return 0;
}
/*
输出:
Global print function
*/

这种情况下,同名的 print 函数可能引发歧义,导致意外的错误行为。


2. 什么是命名空间(namespace)?

2.1 定义

命名空间是 C++ 提供的一种机制,用来组织代码,解决名字冲突问题。
通过命名空间,程序员可以为变量、函数或类添加“所属空间”的限定,避免与其他模块中的名字冲突。

2.2 语法格式

namespace [命名空间的名字]{// 定义变量、函数、类
}
  • namespace_name 是命名空间的名字。
  • 命名空间的成员通过 namespace_name::member 访问。

3. 使用命名空间解决名字冲突

3.1 基本使用

将同名的函数或变量放入不同的命名空间,可以解决名字冲突问题:

#include <iostream>
using namespace std;namespace ModuleA {void print() {cout << "ModuleA's print function" << endl;}
}namespace ModuleB {void print() {cout << "ModuleB's print function" << endl;}
}int main() {ModuleA::print();  // 调用 ModuleA 的 print 函数ModuleB::print();  // 调用 ModuleB 的 print 函数return 0;
}
/*
输出:
ModuleA's print function 
ModuleB's print function
*/

解释:

  • print() 函数分别放在 ModuleAModuleB 命名空间中。
  • 通过 ModuleA::print()ModuleB::print() 调用,避免了名字冲突。

3.2 嵌套命名空间

命名空间可以嵌套使用,用于组织更复杂的代码:

#include <iostream>
using namespace std;namespace Outer {namespace Inner {void print() {cout << "Inner namespace print function" << endl;}}
}int main() {Outer::Inner::print();  // 调用嵌套命名空间中的函数return 0;
}
/*
输出:
Inner namespace print function
*/

注意:

  • 嵌套命名空间使代码结构更清晰,但调用成员时需要完整的命名空间路径。

4. 使用命名空间中的成员

4.1 使用 using 声明

通过 using namespace 声明,可以简化命名空间成员的调用:

#include <iostream>
using namespace std;namespace ModuleA {void print() {cout << "ModuleA's print function" << endl;}
}int main() {using namespace ModuleA;  // 引入 ModuleA 命名空间print();  // 直接调用 ModuleA 的 print 函数return 0;
}
/*
输出:
ModuleA's print function
*/

4.2 注意点

  1. 如果多个命名空间中有同名成员,不能直接使用 using namespace,需要明确调用:

    using namespace ModuleA;
    using namespace ModuleB;
    print();  // 错误:不明确调用哪个命名空间的 print
    
  2. 可以使用 using 声明单个成员:

    using ModuleA::print;  // 只引入 ModuleA 的 print 函数
    print();
    

5. 无命名空间的情况

5.1.问题

在没有命名空间的情况下,同名冲突会导致程序错误。例如:

#include <iostream>
using namespace std;void print() {cout << "Global print function" << endl;
}void print() {  // 重复定义cout << "Duplicate print function" << endl;
}int main() {print();  // 错误:重复定义导致编译错误return 0;
}

5.2.解决

将函数放入不同的命名空间即可避免冲突。


6. 标准命名空间 std

6.1 定义

C++ 标准库中的所有内容都定义在命名空间 std 中,例如 coutcinstring 等。

6.2使用方法

  • 通过 std:: 使用标准库成员:

    std::cout << "Hello, World!" << std::endl;
    
  • 或者引入 std 命名空间:

    using namespace std;
    cout << "Hello, World!" << endl;
    

6.3注意:

  • 在大型项目中,避免使用 using namespace std;,因为可能会与用户定义的名字冲突。

7. 总结

  1. 同名冲突:

    • 不同模块可能定义相同名字的变量、函数或类,导致冲突。
  2. 命名空间:

    • 通过 namespace 将代码分组,解决名字冲突。
    • 成员访问语法:namespace_name::member
  3. using 的使用:

    • 使用 using namespace 简化命名空间成员的调用,但要注意避免歧义。
  4. 标准命名空间:

    • std 是 C++ 标准库的命名空间,包含 coutcin 等标准成员。


三、总结

1.异常处理

  1. 异常处理通过 try-catch 机制捕获和处理异常,防止程序崩溃。
  2. 使用 throw 抛出异常,catch 捕获异常。
  3. 可自定义异常类,用于处理复杂错误。

2.命名空间

  1. 命名空间用于组织代码,避免命名冲突。
  2. 使用 namespace_name::identifier 访问命名空间中的成员。
  3. 可以使用 using namespace 简化调用。

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

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

相关文章

Dynamics 365 CRM- 后端

Dynamics 365 CRM 后端插件语法示例 public IPluginExecutionContext context null;//上下文 public IOrganizationServiceFactory serviceFactory null;//组织服务工厂对象 public IOrganizationService service null;//Org服务对象//创建执行上下文 context (IPluginExe…

C语言——实现并求出两个数的最大公约数

问题描述&#xff1a;求出两个数的最大公约数 //求两个数的最大公约数 #include<stdio.h> #include<stdlib.h> #include<string.h> #include<math.h> #include<time.h>int main() {int a,b;printf("请您输入两个数 a 和 b\n");scanf…

采用qL-MPC技术进行小型固定翼无人机的路径跟随控制

来自论文"Predictive Path-Following Control for Fixed-Wing UAVs Using the qLMPC Framework in the Presence of Wind Disturbances" 控制架构 采用的是 ULTRA-Extra无人机&#xff0c;相关参数如下&#xff1a; 这里用于guidance law的无人机运动学模型为&#…

三维无人机航迹算法的目标函数如何确定

一、定义目标函数 在三维无人机航迹算法中,目标函数的确定通常基于具体的任务需求和飞行约束。以下是一个简单的例子,展示了如何为三维无人机航迹规划定义一个目标函数。 例子:最小化飞行时间和避障的三维无人机航迹规划 1.任务描述:无人机需要从起点飞到终点,同时避开一些…

《Java核心技术I》Swing用户界面组件

Swing和模型-视图-控制器设计模式 用户界面组件各个组成部分&#xff0c;如按钮&#xff0c;复选框&#xff0c;文本框或复杂的树控件&#xff0c;每个组件都有三个特征&#xff1a; 内容&#xff0c;如按钮的状态&#xff0c;文本域中的文本。外观&#xff0c;颜色&#xff0c…

【Office】Office实现shift+鼠标滚轮左右滑动

Office实现shift鼠标滚轮左右滑动 windows系统安装office之后发现&#xff0c;使用shift鼠标滚轮不能够实现左右滑动&#xff0c;我记得以前的office好像是可以的&#xff0c;然后在网上找了一下&#xff0c;找到了一个插件可以实现这个功能 OfficeScroll插件 下载地址&…

vlan和vlanif

文章目录 1、为什么会有vlan的存在2、vlan(虚拟局域网)1、vlan原理1. 为什么这样划分了2、如何实现不同交换机相同的vlan实现互访呢3、最优化的解决方法&#xff0c;vlan不同交换机4、vlan标签和vlan数据帧 5、vlan实现2、基于vlan的划分方式1、基于接口的vlan划分方式2、基于m…

Web项目图片视频加载缓慢/首屏加载白屏

Web项目图片视频加载缓慢/首屏加载白屏 文章目录 Web项目图片视频加载缓慢/首屏加载白屏一、原因二、 解决方案2.1、 图片和视频的优化2.1.1、压缩图片或视频2.1.2、 选择合适的图片或视频格式2.1.3、 使用图片或视频 CDN 加速2.1.4、Nginx中开启gzip 三、压缩工具推荐 一、原因…

【CAN模块】介绍一种检查CAN模块芯片好坏的方法(SN65HVD230)

文章目录 前言一、以SN65HVD230为例介绍端口特性二、代码实现总结 前言 CAN总线收发器&#xff0c;是CAN控制器和物理总线间的接口器件&#xff0c;通常工程师会按照底层协议对其控制&#xff0c;近日笔者仔细了解了CAN总线收发器的物理原理&#xff0c;找到了一种通过观察端口…

RTMP推流平台EasyDSS在无人机推流直播安防监控中的创新应用

无人机与低空经济的关系密切&#xff0c;并且正在快速发展。2024年中国低空经济行业市场规模达到5800亿元&#xff0c;其中低空制造产业占整个低空经济产业的88%。预计未来五年复合增速将达到16.03%。 随着科技的飞速发展&#xff0c;公共安防关乎每一个市民的生命财产安全。在…

[win10] win10系统的下载及在虚拟机中详细安装过程(附有下载文件)

前言 win10 下载&#xff1a;https://pan.quark.cn/s/eb40e8ca57fb 提取码&#xff1a;VTZq 失效&#xff08;可能被官方和谐&#xff09;可评论或私信我重发 下载压缩包后解压 &#xff01;&#xff01;安装路径不要有中文 解压下载的.zip文件&#xff0c;得到.iso文件 打开…

lightRAG 论文阅读笔记

论文原文 https://arxiv.org/pdf/2410.05779v1 这里我先说一下自己的感受&#xff0c;这篇论文整体看下来&#xff0c;没有太多惊艳的地方。核心就是利用知识图谱&#xff0c;通过模型对文档抽取实体和关系。 然后基于此来构建查询。核心问题还是在解决知识之间的连接问题。 论…

[代码随想录17]二叉树之最大二叉树、合并二叉树、二搜索树中的搜索、验证二叉搜索树。

前言 二叉树的题目还是要会一流程构造函数之类的。其中还有回溯的思想 题目链接 654. 最大二叉树 - 力扣&#xff08;LeetCode&#xff09; 一、最大二叉树 思路&#xff1a;还是考察构造二叉树&#xff0c;简单来说就是给你一个数组去构建一个二叉树&#xff0c;递归来解决就…

Docker概述与基础入门

1. 什么是Docker&#xff1f; Docker 是一个开源的平台&#xff0c;用于自动化应用程序的构建、部署和管理。它允许开发人员通过将应用程序及其依赖项打包成容器镜像&#xff0c;从而确保应用可以在任何环境中一致地运行。Docker 容器是轻量级的、可移植的、且具有高度隔离性的…

C# 探险之旅:第三十六节 - 类型class之密封类Sealed Classes

嗨&#xff0c;探险家们&#xff01;欢迎再次搭乘我们的C#魔法列车&#xff0c;今天我们要去一个神秘又有点“傲娇”的地方——密封类&#xff08;Sealed Classes&#xff09;领地。系好安全带&#xff0c;咱们要深入“密封”的奇妙世界啦&#xff01; 什么是密封类&#xff1…

QTreeView 与 QTreeWidget 例子

1. 先举个例子 1班有3个学生&#xff1a;张三、李四、王五 4个学生属性&#xff1a;语文 数学 英语 性别。 语文 数学 英语使用QDoubleSpinBox* 编辑&#xff0c;范围为0到100,1位小数 性别使用QComboBox* 编辑&#xff0c;选项为&#xff1a;男、女 实现效果&#xff1a; 2…

UE5 C++ Subsystem 和 多线程

一.Subsystem先做一个简单的介绍&#xff0c;其实可以去看大钊的文章有一篇专门讲这个的。 GamePlay框架基础上的一个增强功能&#xff0c;属于GamePlay架构的范围。Subsystems是一套可以定义自动实例化和释放的类的框架。这个框架允许你从5类里选择一个来定义子类(只能在C定义…

Linux 添加spi-nor flash支持

1. spi-nor flash简介 在嵌入式ARM开发过程中通常会使用到spi-nor flash&#xff0c;主要用于固化u-boot镜像以支持spi方式启动系统。目前常用的spi-nor flash有gd25wq128e、w25q128等flash芯片&#xff0c;下述以gd25wq128e为例进行讲解。 2.调试通常遇到的问题 无法识别到…

C# 探险之旅:第三十七节 - 类型class之Object:万物之源的奇妙冒险

嘿&#xff0c;勇敢的探险家们&#xff01;欢迎再次踏上C#的神秘之旅。今天&#xff0c;我们将深入探索一个极其强大又无处不在的“大佬”——Object 类型。想象一下&#xff0c;它就像是C#世界里的“超级英雄祖先”&#xff0c;几乎所有的类型都得叫它一声“老祖宗”。 Objec…

LabVIEW实验站反馈控制系统

开发了一套基于LabVIEW的软X射线磁性圆二色实验站的反馈控制系统。这套系统主要用于实现对实验站高电压的精确控制&#xff0c;从而保持照射在样品上的流强稳定性&#xff0c;为分析样品吸收谱提供可靠基准&#xff0c;同时提供了易用的用户界面和强大的数据存储功能。 项目背景…