C++ try/catch/throw

本文通过一个简单的 C++ 程序示例介绍了 try/catch/throw 的使用方法,展示了如何处理除数为零的情况,并解释了 catch 和 throw 数据类型的匹配原则。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

转载自  http://blog.sina.com.cn/s/blog_a9303fd901018ost.html

首先通过一个简单的例子来熟悉C++ 的 try/catch/throw:

   #include<iostream.h>                                 //包含头文件
   #include<stdlib.h>
   double fuc(double x, double y)                        //定义函数
   {
       if(y==0)
       {
           throw y                                   //除数为0,抛出异常
       }
       return x/y;                                    //否则返回两个数的商
10    }
11    void main()
12    {
13        double res;
14        try                                            //定义异常
15        {
16            res=fuc(2,3);
17            cout<<"The result of x/y is : "<<res<<endl;
18            res=fuc(4,0);                                //出现异常
19        }
20        catch(double)                                    //捕获并处理异常
21        {
22            cerr<<"error of dividing zero.\n";
23            exit(1);                                    //异常退出程序
24        }
25    }

catch 的数据类型需要与throw出来的数据类型相匹配的。

在使用 `try/catch` 时遇到问题,通常与异常捕获的范围、类型以及语言特性有关。以下是一些常见问题及其解决方案: ### 1. 捕获系统级异常(如段错误、除零等) 在 C++ 中,标准的 `try/catch` 只能捕获通过 `throw` 抛出的异常,并不能处理系统级异常,例如访问非法内存地址或除以零等操作。为了解决这类问题,可以使用 Windows 特有的 **结构化异常处理**(SEH),即 `__try/__except` 块来捕获硬件异常[^1]。 ```cpp #include <windows.h> __try { // 可能引发异常的代码 int a = 5 / 0; } __except (EXCEPTION_EXECUTE_HANDLER) { printf("捕获到除零异常\n"); } ``` 对于 Qt 程序员,虽然 `QT_TRY/QT_CATCH` 是兼容 SEH 的封装,但在某些情况下仍无法替代原生 SEH 处理机制。 ### 2. PHP 中无法捕获致命错误 PHP 的 `try/catch` 在早期版本中仅能捕获 `Exception` 类型的异常,而无法捕获致命错误(如 `E_ERROR`)。从 PHP 7 开始,很多致命错误被转换为 `Error` 实例,它们实现了 `Throwable` 接口,因此可以通过如下方式统一捕获: ```php try { // 可能抛出异常或错误的代码 } catch (\Throwable $t) { echo "捕获到异常或错误:" . $t->getMessage(); } ``` 此外,若设置了自定义错误处理器(如 `set_error_handler()`),但未将其转换为异常,则这些错误也不会进入 `try/catch` 流程中。 ### 3. Java 中局部变量作用域问题 在 Java 中,如果在 `try` 块内部声明变量,外部将无法访问该变量。为避免此类编译错误,应在 `try/catch` 外部声明变量,并在块内进行赋值: ```java public void test() { String result = null; try { result = JsonUtils.object2Json(map); } catch (IOException e) { e.printStackTrace(); } System.out.println(result); // 正常引用 } ``` 此做法确保了变量的作用域足够宽泛,可被后续代码访问[^3]。 ### 4. JavaScript Promise 异常捕获失败 在异步编程中,直接对 `Promise.reject()` 使用 `try/catch` 是无效的,因为 `Promise` 错误是异步触发的。应使用 `.catch()` 方法或者 `await` 配合 `try/catch` 来正确捕获异常: ```javascript async function f2() { try { await new Promise((resolve, reject) => { reject('出错了'); }); } catch (e) { console.log(e); // 成功捕获 } } f2(); ``` 如果不使用 `await`,则需通过链式调用 `.catch()` 来处理错误: ```javascript function f2() { Promise.reject('出错了') .catch(e => console.log(e)); // 捕获并处理错误 } ``` ### 5. Spring Boot Service 层数据库异常捕获失效 当将数据库操作逻辑从 Controller 层移动至 Service 层后,可能出现异常捕获失效的问题。这通常是由于事务管理配置不当导致的。Spring 默认不会将异常传播到上层,除非明确声明事务回滚策略。 解决方法包括: - 使用 `@Transactional(rollbackFor = Exception.class)` 注解,强制事务在所有异常下回滚。 - 明确抛出受检异常或运行时异常,确保异常能够穿透事务代理。 - 若需要自定义异常处理逻辑,建议结合 `@ControllerAdvice` 或 `@ExceptionHandler` 进行全局异常管理。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值