C++基础6:异常

为什么需要异常?

  • 异常机制的处理原理
    程序会出现错误,尤其是不易察觉的错误。需要了解并解决这些错误。通常,程序出现错误,都会强制退出,很难排除错误原因。

C语言如何表示错误

  1. 函数返回值
    • 通常,成功返回0,返回值-1
    • 返回值为指针类型,成功返回非NULL,失败返回值NULL
      例如:malloc();例外shmat()失败返回值为MAP_INVALD(-1)
    • 其它另类的返回值
      fread()/fwrite()返回读写字符长度size_t,超出长度表示失败。
  2. 全局变量errno

异常处理特点

异常提供一个错误专用通道。
优点:

  1. 不干扰正常的返回值。
  2. 必须处理异常。

案例

通过命令行计算两个数字相除。

#include <iostream>
#include <sstream>
using namespace std;

int main(int argc,char* argv[]){
    istringstream iss(argv[1]); // 读取第一个数字
    int a(0);
    iss >> a;
    iss = argv[2]; // 读取第二个数字
    int b(0);
    iss >> b;
    cout<< a/b << endl;// 输出数字相除
}

语法

异常分为两个部分:抛出异常与捕获并处理异常。

  • 抛出异常
throw 表达式;
  • 捕获并处理异常
try {  
        // 保护代码 包含可能抛出异常的语句;  
} catch (类型名 [形参名]) {  
        // catch块 处理异常
}  

特点

  1. 只要抛出异常,异常后的代码不再执行。
  2. 异常的所抛出与经过的栈都会销毁。

异常机制try-throw-catch的目标是问题检测与问题解决分离

复杂一点地写法

try {  
        // 保护代码 包含可能抛出异常的语句;  
} catch (类型名1 [形参名]) {  
        // catch块 处理异常
} catch (类型名2 [形参名]) {  
        // catch块 处理异常
} catch (类型名3 [形参名]) {  
        // catch块 处理异常
} catch(...){
        // catch块 处理异常
}

注意

  • 异常捕获具有类型匹配,只有相同的或者父类类型才能匹配到。
  • 如果多个catch都能接受相同异常,只有最前面的一个可以接收到。
    catch(...)只能放在所有异常捕获的最后

异常的接口声明/异常规范

返回值类型 函数() throw(异常列表);

指定函数可以抛出何种异常,如果没有throw(异常列表)默认可以抛出所有异常。
指定函数不抛出函数,异常列表为空throw()

那么当异常抛出后新对象如何释放?

异常处理机制保证:异常抛出的新对象并非创建在函数栈上,而是创建在专用的异常栈上,因此它才可以跨接多个函数而传递到上层,否则在栈清空的过程中就会被销毁。所有从trythrow语句之间构造起来的对象的析构函数将被自动调用。但如果一直上溯到main函数后还没有找到匹配的catch块,那么系统调用terminate()终止整个程序,这种情况下不能保证所有局部对象会被正确地销毁。

举例

  1. 捕获异常
#include <iostream>
using namespace std;

void test(){
    cout << "before throw." << endl;
    throw -1;
    cout << "after throw." << endl;
}
int main(){
    try{
        test();
    }catch(int a){
        cout << "exception:" << a << endl;
    }
}
  1. 异常与局部对象析构
#include <iostream>
using namespace std;

class Test{
public:
    Test(){
        cout << "Test Init" <<endl;
    }
    ~Test(){
        cout << "Test Destroy" <<endl;
    }
};

int main(){
    try{
        Test t;
        cout << "before throw." << endl;
        throw -1;
        cout << "after throw." << endl;
    }catch(int a){
        cout << "exception:" << a << endl;
    }
}

注意事项

  1. 如果抛出的异常一直没有函数捕获(catch),则会一直上传到c++运行系统那里,导致整个程序的终止。
  2. 一般在异常抛出后资源可以正常被释放,但注意如果在类的构造函数中抛出异常,系统是不会调用它的析构函数的,处理方法是:如果在构造函数中要抛出异常,则在抛出前要记得删除申请的资源。
  3. 异常处理仅仅通过类型而不是通过值来(switch-case)匹配的,所以catch块的参数可以没有参数名称,只需要参数类型。
  4. 函数原型中的异常说明要与实现中的异常说明一致,否则容易引起异常冲突。
  5. 应该在throw语句后写上异常对象时,throw先通过Copy构造函数构造一个新对象,再把该新对象传递给 catch.
  6. catch块的参数推荐采用地址传递而不是值传递,不仅可以提高效率,还可以利用对象的多态性。另外,派生类的异常扑获要放到父类异常扑获的前面,否则,派生类的异常无法被扑获。
  7. 编写异常说明时,要确保派生类成员函数的异常说明和基类成员函数的异常说明一致,即派生类改写的虚函数的异常说明至少要和对应的基类虚函数的异常说明相同,甚至更加严格,更特殊。

标准异常类

exception派生

异常类 作用
logic_error 逻辑错误,在程序运行前可以检测出来
runtime_error 运行时错误,仅在程序运行中检测到

逻辑异常logic_error派生

异常类 作用
domain_error 违反了前置条件
invalid_argument 指出函数的一个无效参数
length_error 指出有一个超过类型size_t的最大可表现值长度的对象的企图
out_of_range 参数越界
bad_cast 在运行时类型识别中有一个无效的dynamic_cast表达式
bad_typeid 报告在表达试typeid(*p)中有一个空指针p

运行时runtime_error派生

异常类 作用
range_error 违反后置条件
bad_alloc 存储分配错误

尝试捕获逻辑异常和运行时异常

自定义异常类

  • 编码流程
    1.继承异常类exception
    2.实现接口what()
  • 代码结构
class 异常类:public exception {    
public:    
   const char* what()const throw() {
        return 信息字符串;    
   }        
}; 

构造函数、析构函数的异常处理

  • 构造函数可以抛出异常,此时不会调用析构函数,所以如果抛出异常前,申请了资源,需要自己释放。
  • C++标准指明析构函数不能、也不应该抛出异常。
  • C++标准规定,构造函数失败,析构函数不会执行。就是说在构造函数抛出异常前分配的资源将无法释放。
  1. 如果析构函数抛出异常,则异常点之后的程序不会执行,如果析构函数在异常点之后执行了某些必要的动作比如释放某些资源,则这些动作不会执行,会造成诸如资源泄漏的问题。
  2. 通常异常发生时,c++的机制会调用已经构造对象的析构函数来释放资源,此时若析构函数本身也抛出异常,则前一个异常尚未处理,又有新的异常,会造成程序崩溃的问题。

是否使用异常机制

为什么很多经典书籍鼓励使用异常,但是实际开发中很多C++编码规范却禁用异常?
C++异常机制在语法上是更加优雅的处理错误,但是实际上编译出来的程序会有一些性能损失,另外错误地使用异常处理代码会变得更加复杂。

编译器选项

g++特殊编译选项g++ -fno-exceptions

在不同的编码规范中,对是否使用异常存在争议。

C语言实现异常机制

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,686评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,668评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,160评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,736评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,847评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,043评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,129评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,872评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,318评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,645评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,777评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,470评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,126评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,861评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,095评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,589评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,687评论 2 351

推荐阅读更多精彩内容