尧图网站建设 尧图网络
  • 首页
  • 关于我们
  • 服务项目
  • 案例展示
  • 建站流程
  • 资讯中心
  • 联系我们
首页/资讯中心/详情

【C++】零基础入门 · 第 13 节:异常处理(try、catch、throw)

【C++】零基础入门 · 第 13 节:异常处理(try、catch、throw)
📅 发布时间:2026/6/24 17:15:30

在前面 12 节中,我们学习了变量、函数、类、指针、文件操作、模板和 STL。这些都是「怎么写代码」的知识。今天,我们来学习一个同样重要但经常被初学者忽略的主题——异常处理。它解决的是「代码出错了怎么办」的问题。

1. 为什么需要异常处理?

程序运行时,总会遇到各种意外情况:文件找不到、内存不够用、数组越界、除以零……这些情况如果不处理,程序就会直接崩溃。

你可能会说:用if判断一下不就行了?确实可以,但if有一个局限——它只能处理「当前函数」里的错误。如果错误发生在深层嵌套的函数调用中,你需要一层一层地把错误信息往上传,非常麻烦。

异常处理提供了一种更优雅的方式:在出错的地方「抛出」异常,在合适的地方「捕获」它。错误信息会自动沿着调用链向上传递,直到被处理。

2. 基本语法:try、catch、throw

C++ 的异常处理由三个关键字组成:

  • try:包裹可能出错的代码
  • throw:当检测到错误时,「抛出」一个异常
  • catch:「捕获」异常并处理

2.1 最简单的例子

#include<iostream>usingnamespacestd;intdivide(inta,intb){if(b==0){throw"除数不能为零!";// 抛出异常}returna/b;}intmain(){try{cout<<divide(10,2)<<endl;// 正常执行cout<<divide(10,0)<<endl;// 会触发异常cout<<"这行不会被执行"<<endl;// 异常后跳过}catch(constchar*msg){cout<<"捕获到异常:"<<msg<<endl;}cout<<"程序继续正常运行"<<endl;return0;}

输出:

5 捕获到异常:除数不能为零! 程序继续正常运行

几个关键点:

  • throw抛出异常后,try块中剩余的代码不会被执行,程序直接跳转到对应的catch块。
  • catch处理完异常后,程序继续往下执行,不会崩溃。
  • throw后面可以跟任何类型的值:字符串、整数、自定义对象等。

2.2 多个 catch 块

你可以针对不同类型的异常写不同的处理逻辑:

#include<iostream>usingnamespacestd;voidprocess(inttype){if(type==1)throw42;if(type==2)throw"出错了";if(type==3)throw3.14;}intmain(){for(inti=1;i<=3;i++){try{process(i);}catch(inte){cout<<"整数异常:"<<e<<endl;}catch(constchar*e){cout<<"字符串异常:"<<e<<endl;}catch(doublee){cout<<"浮点异常:"<<e<<endl;}}return0;}

输出:

整数异常:42 字符串异常:出错了 浮点异常:3.14

catch块会按照书写的顺序匹配异常类型,只会执行第一个匹配的catch块。

3. 标准异常类

在实际开发中,我们通常不会throw裸字符串或整数,而是使用 C++ 标准库提供的异常类。它们都在<stdexcept>头文件中。

3.1 常见的标准异常

异常类用途
std::runtime_error运行时错误(如文件不存在)
std::invalid_argument无效参数
std::out_of_range越界访问
std::overflow_error算术溢出
std::logic_error逻辑错误

3.2 使用标准异常

#include<iostream>#include<stdexcept>#include<vector>usingnamespacestd;intgetElement(constvector<int>&vec,intindex){if(index<0||index>=vec.size()){throwout_of_range("下标 "+to_string(index)+" 越界!");}returnvec[index];}intmain(){vector<int>nums={10,20,30};try{cout<<getElement(nums,1)<<endl;// 20cout<<getElement(nums,5)<<endl;// 越界!}catch(constout_of_range&e){cout<<"异常:"<<e.what()<<endl;}return0;}

输出:

20 异常:下标 5 越界!

标准异常类都有一个what()方法,返回错误描述信息。用const 引用(const exception&)来捕获是一个好习惯,可以避免不必要的拷贝。

3.3 统一捕获所有异常

如果不确定会抛出什么类型的异常,可以用catch (...)捕获所有异常:

try{// 可能出错的代码}catch(constexception&e){cout<<"标准异常:"<<e.what()<<endl;}catch(...){cout<<"未知异常"<<endl;}

建议把catch (const exception&)放在前面,catch (...)放在最后作为兜底。

4. 自定义异常类

当标准异常类不能满足需求时,你可以定义自己的异常类。通常的做法是继承std::exception:

#include<iostream>#include<exception>#include<string>usingnamespacestd;classMyException:publicexception{private:string message;public:MyException(conststring&msg):message(msg){}constchar*what()constnoexceptoverride{returnmessage.c_str();}};voidriskyOperation(){throwMyException("自定义异常:操作失败!");}intmain(){try{riskyOperation();}catch(constMyException&e){cout<<e.what()<<endl;}return0;}

输出:

自定义异常:操作失败!

自定义异常的好处是你可以携带更多的上下文信息(比如错误代码、发生位置等),方便调试和日志记录。

5. 异常的传播机制

理解异常是怎么「传递」的,对于正确使用异常处理至关重要。

5.1 调用链中的异常传播

#include<iostream>#include<stdexcept>usingnamespacestd;voidfuncC(){throwruntime_error("funcC 中出错了");}voidfuncB(){funcC();// 不处理,继续向上传}voidfuncA(){try{funcB();// 不处理,继续向上传}catch(construntime_error&e){cout<<"funcA 捕获:"<<e.what()<<endl;}}intmain(){funcA();return0;}

输出:

funcA 捕获:funcC 中出错了

异常从funcC抛出,经过funcB(没有catch),最终在funcA中被捕获。异常会沿着调用链自动向上传递,直到找到匹配的catch块。

5.2 没有被捕获的异常

如果异常一路传到main函数都没有被捕获,程序会调用std::terminate()直接终止,并可能弹出系统级的错误提示。所以一定要确保所有可能抛出异常的地方都有对应的catch处理。

6. RAII:C++ 资源管理的黄金法则

异常处理有一个容易被忽视的问题:资源泄漏。如果在try块中申请了内存或打开了文件,异常发生后这些资源可能不会被释放。

C++ 的解决方案是RAII(Resource Acquisition Is Initialization)——把资源的生命周期绑定到对象的生命周期上。当对象离开作用域时,析构函数会自动释放资源,即使是因为异常导致的离开。

#include<iostream>#include<fstream>#include<stdexcept>usingnamespacestd;voidprocessFile(conststring&filename){ifstreamfile(filename);// RAII:文件在构造时打开if(!file.is_open()){throwruntime_error("无法打开文件:"+filename);}string line;while(getline(file,line)){cout<<line<<endl;}// 函数结束时,file 的析构函数自动关闭文件// 即使中途抛出异常,析构函数也会被调用}intmain(){try{processFile("test.txt");}catch(construntime_error&e){cout<<"错误:"<<e.what()<<endl;}return0;}

STL 容器(vector、string等)和智能指针(unique_ptr、shared_ptr)都遵循 RAII 原则。在 C++ 中,优先使用 RAII 管理资源,而不是手动new/delete或open/close。

7. 使用异常的最佳实践

7.1 什么时候该用异常

  • 真正的异常情况:文件不存在、网络断开、内存不足等不可预期的错误
  • 不适合正常流程控制:不要用异常来代替if-else判断

7.2 异常安全的三个级别

级别保证说明
基本保证程序不会泄漏资源最低要求
强保证操作要么完全成功,要么回到操作前的状态事务性
不抛出保证函数保证不抛出异常用noexcept声明

7.3 使用noexcept标记不抛出异常的函数

如果你确定某个函数不会抛出异常,可以用noexcept声明,帮助编译器做更好的优化:

intadd(inta,intb)noexcept{returna+b;}

析构函数默认就是noexcept的,你不应该在析构函数中抛出异常。

8. 总结

这一节我们学习了 C++ 的异常处理机制:

  • try包裹可能出错的代码,throw抛出异常,catch捕获并处理。
  • 标准异常类(runtime_error、out_of_range等)提供了统一的错误描述接口。
  • 异常沿着调用链自动向上传递,直到被catch捕获。
  • RAII 是 C++ 资源管理的核心原则,确保异常发生时资源不会泄漏。
  • 异常只用于处理真正的异常情况,不要用来做流程控制。

异常处理是编写健壮程序的重要保障。掌握了它,你的代码就能在面对意外情况时「优雅地失败」,而不是直接崩溃。下一节我们将继续探索 C++ 的更多高级特性。加油!

相关新闻

  • 加油
  • Blender建筑建模革命:用building_tools插件告别繁琐手动建模
  • 5分钟快速上手:跨平台资源下载工具res-downloader终极指南

最新新闻

  • OpenClaw AI协作系统:构建可审计、低延迟的AI工程化工作流
  • 深入Frida源码:从动态插桩原理到Hook执行全流程解析
  • 2025年精选6款漏洞扫描工具:从原理到实战的完整指南
  • GLM-4.7-Flash+MCP:面向开发工作流的结构化AI加速器
  • Firefox Hackbar v2.1.3:HTTP请求构造与Web安全测试实战指南
  • OpenClaw不是QQ机器人,而是服务编排型消息总线

日新闻

  • 终极指南:如何用shadPS4在电脑上免费畅玩PS4游戏
  • 打造个性化Instagram Clone:主题定制与用户体验优化技巧
  • 未来展望:RoseTTAFold-All-Atom的发展路线图与社区支持资源汇总

周新闻

  • Visual C++运行库修复终极指南:5分钟快速解决Windows软件启动错误
  • 手把手教你构建统计局地区经济数据爬虫:从环境搭建到数据持久化全指南
  • 2026多Agent深度解析:用AI团队替代单一模型,四种架构实战落地

月新闻

  • 【总结】入门篇:50句话让你记住架构核心概念
  • WeChatMsg技术方案解析:实现Mac微信数据自主管理的完整解决方案
  • WeChatMsg:革新性微信数据备份方案,打造你的专属数字记忆库

关于尧图

  • 公司简介
  • 团队介绍
  • 企业文化
  • 荣誉资质

服务项目

  • 定制开发
  • 电商建站
  • UI 设计
  • 运维服务

快速链接

  • 案例展示
  • 建站流程
  • 常见问题
  • 资讯中心

联系方式

  • 📍北京市朝阳区互联网产业园 A 座 10 层
  • 📞400-888-8888
  • ✉️contact@rkmt.cn
  • 🕐周一至周日 9:00-21:00

© 2024 北京尧图网络科技有限公司 版权所有 | 京 ICP 备 XXXXXXXX 号