zoukankan      html  css  js  c++  java
  • 【C++】异常简述(二):C++的异常处理机制

      上文简述了在C语言中异常的处理机制,本文主要讲解C++中的异常处理.

    一、异常的语法格式

      在C++中,异常的抛出和处理主要使用了以下三个关键字:try、 throw 、 catch.其格式如下:

      当我们在程序中想抛出一个异常时,可以这样:

    #include<iostream>
    #include<exception>
    using namespace std;
    int Div(int left,int right){
    	if(right==0){
    		throw exception("除数不能为0");
    	}
    	return left/right;
    }
    

      当我们想使用这个函数时,需要在函数外部进行异常的捕获:

    int main(){
    	try{
    		Div(10,20);	//合法
    		Div(10,30);	//合法
    		Div(10,0);	//非法,会抛出异常
    	}catch(exception & e){
    		e.what();	//打印异常信息
    	}
    	return 0;
    }
    

      如果存在不同类型的异常,我们可以这样:

    try{
        //包含可能抛出异常的语句;
    }catch(类型名 [形参名]){
        //可能出现的异常1
    }catch(类型名 [形参名]){
        //可能出现的异常2
    }catch(...){
        //如果不确定异常类型,在这里可以捕获所有类型异常!
    }
    

    二、异常抛出与捕获

      1.异常是通过抛出对象而引发的,该对象的类型决定了应该激活哪部分代码.

      就上述代码来说,我们throw了一个exception对象,因此在捕获异常时,最终会匹配到catch到exception的代码块.

      2.被选中的处理代码是调用链中与该对象类型匹配且离抛出位置最近的那个.

      当try内的代码块出现异常时,系统会根据catch的顺序和参数的匹配程度来选择执行哪个代码块,因此,系统会选择最靠前且参数越匹配的代码块.

      3. 抛出异常后会释放局部存储对象,所以被抛出的对象也就还给系统了,throw表达式会初始化一个抛出特殊的异常对象副本(匿名对象),异常对象由编译管理,异常对象在传给对应的catch处理之后撤销。

      也就是说,在上述的除法代码中,我们throw出的对象在抛出异常后会还给操作系统,而throw表达式会自己初始化一个匿名的对象副本,在传给catch相应的代码块后被回收.

    三、栈展开

      当我们的程序抛出异常时,会先暂停当前函数的执行,开始查找对应匹配的catch语句.

      首先会检查throw是否在代码块内部.

      如果是的话,再去找匹配的catch子句.

        如果有匹配的,则进行处理.

        如果没有,则退出当前函数栈,继续在调用函数的栈中进行从查找.

      不断重复上述过程.

      如果到达main函数栈,依旧没有匹配,则直接终止程序.

      上述沿着调用链查找匹配的catch子句的过程称之为栈展开.

      找到匹配的catch子句并处理以后,会继续沿着catch子句后面继续执行.

    四、异常捕获的匹配规则

      异常对象类型与catch说明符的类型必须完全匹配,只有以下几种情况例外:

      1.允许从非const对象到const类型对象的转换.

      2.允许派生类型到基类类型的转换.

      3.将数组转换为指向数组类型的指针,将函数转换为指向函数的指针.

    五、异常规范

      在函数声明之后,列出该函数可能抛出异常类型,并保证该函数不会抛出其他类型的异常.

      1、成员函数在类内声明和类外定义两处必须有相同的异常规范.

      2、函数抛出一个没有被列在它异常规范中的异常时(且函数中抛出异常没有在函数内部进行处理),系统调用C++标准库中定义的函数unexpected().

      3、如果异常规范为throw(),则表示不得抛出任何异常,该函数不用放在try块中.

      4、派生类的虚函数的异常规范必须与基类虚函数的异常规范一样或更严格(是基类虚函数的异常的子集).

      (因为派生类的虚函数被指向基类类型的指针调用时,保证不会违背基类成员函数的异常规范.)

    六、异常与构造&析构函数

      1. 构造函数完成对象的构造和初始化,需要保证不要在构造函数中抛出异常,否则可能导致对象不完整或没有完全初始化.

      2. 析构函数主要完成资源的清理,需要保证不要在析构函数内抛出异常,否则可能导致资源泄漏(内存泄漏、句柄未关闭等).

    七、自定义异常类型

      自定义异常类型代码:

    class Exception : public exception{
    public:
    	Exception(int errId = 0,const char* errMsg = "")
    		:_errId(errId)
    		,_errMsg(errMsg){}
    public:
    	virtual const char* what()const{
    		cout<<"errId:"<<_errId<<endl;
    		cout<<"errMsg:"<<_errMsg.c_str()<<endl;
    		return _errMsg.c_str();
    	}
    private:
    	int _errId;
    	string _errMsg;
    }
    

      测试代码:

    void TestException(){
    	throw Exception(1,"错误!");
    }
    int main(){
    	try{
    		TestException();
    	}catch(exception &e){
    		e.what();
    	}
    }
    

      关于C++异常处理这一块的知识暂时先整理到这里,感谢各位读者支持!

  • 相关阅读:
    JSP指令用来设置整个JSP页面相关的属性
    JSP 生命周期 理解JSP底层功能的关键就是去理解它们所遵守的生命周期
    JSP 开发环境搭建
    JSP(Java Server Pages,即:Java服务器页面
    JSP 国际化
    JSP 调试
    JSP 异常处理
    JSP 自定义标签
    JSP JavaBean
    JSP 标准标签库(JSTL)
  • 原文地址:https://www.cnblogs.com/qq329914874/p/6739592.html
Copyright © 2011-2022 走看看