C++常见异常处理原理及代码示例解析

编程中常见的错误

  • 程序的编译错误——比较好解决,主要是一些语法错误
  • 程序的运行错误——产生因素较为复杂,如空间不够,下标越界,访问非法空间等。

异常是指程序运行时出现的不正常,可分为一下几类:

CPU异常;如在计算过程中,出现除数为0的情况。

内存异常,如:

  • 使用new或malloc申请动态内存但存储空间不够;
  • 数组下标越界;
  • 使用野指针、迷途指针读取内存;

设备异常,如:

  • 无法打开文件,或文件损坏;
  • 正在读取磁盘文件时挪动了文件或磁盘;
  • 正在使用打印机但设备被断开;
  • 正在使用的网络断线或阻塞;

用户数据异常,如:

  • scanf输入数据格式或类型有错误;
  • 正在处理的数据库有错误;
  • 程序假定的数据环境发生变化;
  • 异常处理机制

抛出异常(throw)、检查异常(try块)、捕获异常(catch块)

C++是根据类型区分不同异常的,因此在抛出异常时,throw表达式的值没有实际意义,而表达式的类型则非常重要;如果程序中有多处要抛出的异常,应该用不同的表达式类型来相互区别。

关于throw的说明

  • 执行throw的时候,不会执行跟在throw后面的语句,而是将程序从throw转移到匹配的catch,该catch可以是同一函数中的catch,也可以在直接或间接调用发生异常函数的上一级函数中。
  • 被抛出的对象是一个用throw表达式初始化的「异常对象」,异常对象由throw创建,并初始化为被抛出的表达式副本,异常对象将传递给对应的catch,并在异常处理完成后撤销。因此异常对象必须是可以复制的类型(具有复制构造函数)。
  • 如果抛出的是数组,被抛出的对象自动转换为指向该数组首元素的指针,如果抛出的是一个函数,函数被转换为指向该函数的指针。
  • 如果抛出一个指向派生类对象的基类指针,则其对象将被分割,只抛出基类的部分。
  • 抛出指向局部对象的指针总是错误的,因为抛出指针的时候,必须确保进入异常处理程序时,指针所指向的对象仍然存在。

检测捕获异常

一般形式:

try{
  ....//检测程序块(可能抛出异常的代码)
}
catch(异常说明符1){
  ....//处理程序(当异常说明符1被抛出时执行的程序)
}
catch(异常说明符2){
  ....//处理程序(当异常说明符2被抛出时执行的程序)
}
..... //更多的catch

catch子句的形参列表

catch(类型名) //catch只需要了解异常的类型
catch(类型名 形参名) //catch需要了解异常类型之外的信息
catch(...) //捕获所有异常

重抛异常

在catch子句中,可以再次抛出异常,其中throw不加表达式,表示将捕获到的异常再次向上级函数抛出,不会被本函数的其他catch子句捕获。

try{
  throw "hello"; //抛出char* 异常  
}
catch(const char*){	//捕获char*异常
  throw;		//重新抛出char* 异常至上一级函数
}

throw关键字修饰的函数

C++函数后面加关键字throw(something)限制,是对这个函数的异常安全作出限制;这是一种异常规范,只会出现在声明函数时,表示这个函数可能抛出的异常类型。

void fun() throw(); //表示fun函数不允许抛出任何异常,即fun函数是异常安全的
void fun() throw(...); //表示fun函数可以抛出任何形式的异常
void fun() throw(exceptionType) //表示fun函数只能抛出exceptionType类型的异常

如void GetTag() throw(int);表示只能抛出int类型的异常,如果抛出非int类型的异常,则会调用unexsetpion()函数,退出程序。假如在函数声明时用throw()限定(这个函数本身不可能抛出异常),则编译器在决定其优化方式上更加灵活。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持呐喊教程。

声明:本文内容来源于网络,版权归原作者所有,内容由互联网用户自发贡献自行上传,本网站不拥有所有权,未作人工编辑处理,也不承担相关法律责任。如果您发现有涉嫌版权的内容,欢迎发送邮件至:notice#nhooo.com(发邮件时,请将#更换为@)进行举报,并提供相关证据,一经查实,本站将立刻删除涉嫌侵权内容。