C++异常处理
admin
2024-04-08 02:04:10
0

1.传统的C语言错误处理方法



    (1)在函数中返回一个错误信息。比如Linux经常返回-1代表错误。
    也可以设定一个全局的变量,比如errno
    (2)用信号函数signal和raise捕捉信号
    (3)用setjmp和longjmp两个非局部跳转函数,这种方法很困难,耦合度也很高,因为它和goto不一样,他会跳转到其他地方,而不是局部地点。
    以上三种方法都有自己的问题。第一个显得代码过于冗长,第二个信号处理标准不一致就会出很大问题,第三个问题更大,因为它会跳到栈区的其他地方,不会自动调用析构函数,行为危险,结果未知。


2.C++的异常处理方式


2.1 try-throw-catch结构


   try代码块负责运行正常代码,在代码中的错误会通过throw抛出,catch负责接受抛出的错误,做出相应的处理。

(1)throw:抛出异常

#include
#include
//抛出一个异常
//设置一个错误类
class MyError{
private:const char* const data;
public:MyError(const char* const msg = 0):data(msg){}//构造函数};void f()
{throw MyError("something bad happened");
}int main()
{f();
}
//f()函数抛出一个错误,生成了MyError对象,这是程序创建的一个对象的拷贝,f()函数返>回了这个对象。

2.2.捕获异常 catch


catch(type value)

{

        some code       

}


2.3.try块


在try块中编写的代码,所产生的错误都会被根据程序员所设计的throw所抛出,程序会返回到错误开始的地方,如果不对错误做出处理,程序到此就结束了。
    try的使用方法为
    try{
        代码
    }


2.4异常处理器


    处理抛出的异常的地方就叫异常处理器,他会接住throw抛出的异常,异常处理对继承和多态同样有效

使用方法:
    try{
        代码
    }
    catch(类型1 id1)
    {
        处理方法1
    }
    catch(类型2 id2)
    {
        处理方法2
    }
    ……
    异常抛出后,会一一匹配catch,如果类型符合,就能够执行catch块的语句

    异常被抛出以后,会按照顺序匹配异常处理器,一旦找到能够匹配的,就会开始处理,不会继续匹配下一个,即使下面的更加合适。
    匹配一个异常,并不要求异常和处理器之间完全相关,一个对象或者是指向派生类对象的引用都会与其基类处理器匹配。如果不是引用或者指针,而是派生类对象和基类相匹配,
    则会发生截断,派生类对象会被截断,与基类相匹配。

#includeclass X{
public:class Trouble{};class Small:public Trouble{};class Big:public Trouble{};void fun(){throw Big();}};int main(){X x;try{x.fun();}catch(X::Trouble&){std::cout << "捕获到Trouble"<

输出:捕获到Trouble

Small,Big都是Trouble的派生类,抛出Big异常处理的时候,却被第一个异常处理器捕获,不会被第三个捕获。

所以我们在处理异常的时候,可以把基类处理器放在最后面捕获一些未定位的错误。
    注意:异常匹配过程不会发生类型转化,以下面的程序为例子

定义了两个类,Except1和Except2,其中Except2定义了一个自动转换类型的函数。

在抛出异常后,异常处理器接受到抛出的错误类型,会匹配到Except1而不是Except2,它不会自动把Except1转化为Except2。

#include
using std::cout;class Except1{};
class Except2{
public:Except2(const Except1&){}//类型转换,将Except1转换为其他类型
};
void fun()
{throw Except1();//抛出一个异常,类型是Except1
}
int main()
{try{fun();}catch(Except2&){cout << "inside catch(except2)\n";}catch(Except1&){cout << "inside catch(except1)\n";}return 0;
}

运行结过将会是:inside catch(except1)


2.4.1捕获所有的异常


    catch(…)
    {

        some code
    }

    这个表示捕获所有的异常,但是不会接受任何参数,可以把它放在异常处理器的最后面。


2.4.2 不捕获异常


    抛出的异常不能被忽略,需要被捕获,否者就会出错。
    如果没有异常处理器可以捕获抛出的异常,则会进行下面的处理
    (1)terminate()函数
    如果没有可以匹配异常的异常处理器,则会调用这个函数,这个函数调用了C语言标准库中的abort()函数,程序不会被正常终止,不会调用析构函数
    除了异常不会被捕获会调用terminate()函数,还有另外两种情况会调用这个函数
    第一种情况:局部对象的析构函数抛出异常
    第二种情况:静态对象的构造函数或者析构函数抛出异常
    (2)set_terminate()函数
    程序员可以使用set_terminate定义自己的terminate()函数,但是在调用结束后,仍然会调用默认的terminate()函数。
    使用方法如下。

#include
#include
using namespace std;
void terminator()
{cout << "terminator 回来了\n";
}
//设置一个函数指针,指向set_terminator
void (*Myterminator)(void)  = set_terminate(terminator);
class Botch
{
public:class Fruit{};void fun(){cout << "Botch::f()"<

set_terminate函数是一个这样的函数 void set_terminate(void(*func)(void));
其中func是自己定义的
析构函数抛出异常,catch获取了异常,先调用了自己定义的set_terminate()。但是运行结果和我们预料的不一致。

输出:
Botch::f()
terminator 回来了
Aborted (core dumped
)

第一次调用fun函数的时候,确实调用了自定义的terminate函数,但是析构函数所产生的异常却仍然调用了默认的terminate函数,终止了程序。

说明,析构函数抛出的异常,只会调用默认的terminate函数。我们在编写代码的时候,应该避免让构造函数抛出异常。


二.清理


2.1 构造函数抛出异常


有时候异常发生了,我们需要对资源进行清理,比如在构造函数中抛出异常,由于构造函数并没有完成,异常发生后,它不会自动调用析构函数清理资源

#include
using namespace std;class Cat{
public:Cat(){ cout << "cat()" <

 运行结果:
UseResource()
cat()
cat()
cat()
Dog模拟内存分配
n = 44
inside hander

cat的构造函数被调用,Dog却在分配内存的时候抛出一个异常,导致UseResource的构造函数没有结束,所以最后的析构函数没有调用,cat分配的资源仍然没有被释放


2.2管理资源的方法



    为了防止资源泄露,可以使用两种办法来来分配资源
    (1)在构造函数里捕获异常,用于释放资源
    (2)在对象的构造函数分配资源,并且在对象的析构函数中释放资源
  

#include
#include
using namespace std;template
class PWrap
{
private:T* ptr;
public:class RangeError{};//异常类PWrap(){ptr = new T[sz];cout << "PWrap constructor"<= 0 && i < sz)return ptr[i];throw RangeError();}
};class Cat{
public:Cat(){cout << "Cat()" << endl;}~Cat(){cout << "~Cat()"< cats;PWrap dog;};int main()
{try{UseResource ur;}catch(int){cout << "inside handler" << endl;}catch(...){cout << "inside catch(...)" << endl;}return 0;
}


这种方法的好处在于在创建UseResource对象之前,指针类就已经被创建了,所需要创建的对象嵌入到了指针对象中。当Dog出现异常时,cat的析构函数被调用了,就不会有
内存泄露。
输出:
Cat()
Cat()
Cat()
PWrap constructor
动态创建Dog对象
~Cat()
~Cat()
~Cat()
PWrap destructor
inside handler

这样的方法实际上也用于智能指针。

2.3 函数级的try块


#include
using namespace std;
int main()try
{throw "主函数抛出异常";}
catch(const char* msg){cout << msg << endl;}

在函数后面使用try,然后紧跟着函数使用catch接受抛出的异常,这种方法也可以用在类的内部,比如用在构造函数里面。

相关内容

热门资讯

“双标”换卡背后,银行还需多些... 新华社记者 颜之宏、杨深深 持到期银行卡和身份证去银行网点换新卡,却被要求“必须交回旧卡才能取新卡”...
“离境退税2.0”带动“中国购... 【环球时报综合报道】编者的话:5月18日,商务部等6部门联合发布《关于加力优化离境退税措施扩大入境消...
一年烧掉2000亿、市值蒸发3... 商业润点 |Biz Run Review 三国归晋,用了六十年。即时零售的"三国杀",才刚刚开局...
原创 金... 2026年5月22日,国内黄金市场呈现出令人咋舌的价格鸿沟。基础金价徘徊在每克995.3元,而回收价...
原创 人... SpaceX的星舰V3终于在全球瞩目中成功升空。北京时间5月23日清晨,这颗高达124米的巨型火箭顺...
原创 被... 5月19日,欧洲议会掀起了一场引人注目的风暴,以压倒性的票数通过了最新的钢铁进口规定。 这套规则...
光纤量价齐升,烽火通信加快布局... 烽火通信(600498)5月22日披露的投资者关系活动记录表显示,公司于5月21日参加了中国信息通信...
原创 突... 今天5月24日一大早,打开行情一看,国际现货黄金报4508.25美元/盎司,单日跌了26.68美元,...
企业快讯 | 携手联通!狄耐克... 狄耐克 厦门总商会副会长企业 厦门狄耐克智能科技股份有限公司 与中国联通厦门分公司 将5G智慧“嵌入...
美银策略师警告:SpaceX与... 环球网 据彭博社报道,美国银行首席投资策略师迈克尔·哈特奈特(Michael Hartnett)最新...
卸任55天后,知名基金经理任相... 【导读】卸任55天后,知名基金经理任相栋“奔私”谜底揭晓 见习记者 闫军 知名基金经理任相栋“奔私”...
原创 大... “免签+手机刷一切”就能让老外连夜订机票?2026年一季度,阿根廷人来华暴涨九倍,北京三源里菜市场三...
从泰山顶峰掉落!“大佬背后的大... 文/刘工昌 他曾是柳传志的“大哥”,助力联想完成混合所有制改革;是史玉柱眼中的“贵人”,帮他东山再起...
原创 2... 最近网上流传出一份2030年GDP10强预测榜单,其中一些城市位次的变化也挺有趣的。上海排在第一,深...
原创 全... 2026年3月的全球美债市场迎来剧烈变动,彻底打破了长期稳定的持仓格局。 根据美国财政部发布的国际资...
全球都在给这几只“疯牛”烧钱 近段时间,AI行情再次成为全球资本市场主线,但舞台中央的“主角”发生了变化:投资者不再只偏好云厂商和...
【财闻联播】“硬刚监管”?老虎... ★ 宏观动态 ★ 商务部:1—4月全国吸收外资2876.9亿元人民币 据商务部网站,2026年1—4...
燕京啤酒营收净利双增:U8增速... 蓝鲸新闻5月22日讯(记者 朱欣悦)燕京啤酒(000729.SZ)打了一个翻身仗。 2025年燕京啤...
原创 帮... 老铁们,这周有个事儿挺有意思,估计不少基民都看懵了:都说科技是主线,芯片是未来,可数据显示,年内火爆...
4家银行AIC现身存储巨头股东... 近日,资本市场热度颇高的两家存储巨头长鑫科技集团股份有限公司(以下简称“长鑫科技”)、长江存储控股股...