赞
踩
在我们平时写C语言程序时,经常出现会出现一些错误,有些是我们可以预测到的,这些我们可以通过返回错误码,或者设置回调函数打印错误信息等方法处理。但还有些程错误是我们不好预测的,如断错误,这时产生错误我们没有处理就会终止程序,然而我们经常运行程序后遇到错误不希望终止,如我们进程使用一些垃圾软件会闪退,这就是因为他门的程序异常没有处理好,所以遇到预测范围外的错误时,程序就会直接终止。所以为了解决这种情况,在C++里面就提出了异常处理机制,当一个函数无法处理产生的错误时,就抛出异常,让函数的调用者直接或者间接处理从错误。
当 throw 抛出异常时,就会暂停当前函数,先去当前函数里寻找同类型的catch异常捕获,如果当前函数没有,就释放当前函数的栈帧,去上一层函数里去找catch,这里是按照就近原则寻找catch,如果没有找到则退出程序。这个过程称为==栈展开==。
1.代码中抛出异常对象后,会在函数调用链里与该对象类型匹配且离抛出异常处最近的catch匹配(==就近原则==)。
2.异常抛出后,会打乱程序执行流,会释放局部存储对象,因此在调试含异常的代码比较难调试。
3.异常抛出的类型必须和catch捕捉的类型必须匹配才可捕获,只有以下三种情况例外:
4.(==异常的再次抛出==)catch可以将收到的异常处理后再次抛给外层调用链,再由外层catch进一步处理。
5.记住!不要在析构函数里和构造函数里抛异常。分别可能会导致对象数据清理不完全,造成内存泄漏。可能会造成对象初始化不完全,在别人使用时可能调用都野指针。
在写一些项目时,会分工不同的人来合作,那么有三个人来实现三个不同的功能的接口,我要调用他们三个人设计的接口,可是他们都使用了异常,但我又不知道他们抛出的异常都有什么类型,就算知道,在大型项目里也要常设计好多种类型的异常,我难道要每种类型设置一个catch吗?那怎么办呢?
class exception {
public:
exception() throw();
exception(const exception&) throw();
exception& operator= (const exception&) throw();
virtual ~exception() throw();
virtual const char* what() const throw();
}
exception是一个抽象类基类,他们三个人在设计异常对象时可以使异常对象继承这个类,并重写基类里的what函数,what是用来展示异常原因的。那么我调用他们的接口就可以直接catch(exception&e),用基类类型就可以接收所有的子类异常了,通过e.what()就可以展示出异常信息,是不是很方便,如果不明白看下面栗子。
#include<iostream>
#include<string>
#include<windows.h>
using namespace std;
class Exception
{
public:
Exception(const string& msg, int id)
:_errMsg(msg)
, _errId(id)
{}
virtual string what()=0;
protected:
string _errMsg;
int _errId;
};
class connectException : public Exception//连接数据库异常
{
public:
connectException(const string& Msg, int id)
:Exception(Msg , id)
{}
virtual string what()
{
string msg = "【数据库错误:】";
msg += _errMsg;
return msg;
}
};
class getinfoException : public Exception//获取数据异常
{
public:
getinfoException(const string& Msg, int id)
:Exception(Msg, id)
{}
virtual string what()
{
string msg = "【获取数据数据错误:】";
msg += _errMsg;
return msg;
}
};
class echoException : public Exception//响应异常
{
public:
echoException(const string& Msg, int id)
:Exception(Msg, id)
{}
virtual string what()
{
string msg = "【响应错误:】";
msg += _errMsg;
return msg;
}
};
void connectsql()
{
if (rand() % 5 == 0) //这里是模拟函数产生异常,以下雷同
{
throw connectException ("数据库连接错误", 1);
}
if (rand() % 3 == 0)
{
throw connectException ("网络错误",2);
}
cout << "【OK 数据库连接成功】" << endl;
}
void getinfo()
{
if (rand() % 2 == 0)
{
throw getinfoException("获取数据失败", 101);
}
if (rand() % 3 == 0)
{
throw getinfoException("数据处理错误", 102);
}
cout << "【OK 获取客户端信息成功】" << endl;
}
void echoclient()
{
if (rand() % 2 == 0)
{
throw echoException("响应错误", 202);
}
if (rand() % 3 == 0)
{
throw echoException("数据丢失", 203);
}
cout << "【OK 响应客户端成功】" << endl;
}
class HTTPserver
{
public:
void start()
{
while (1)
{
try
{
connectsql();
Sleep(1000);
getinfo();
Sleep(1000);
echoclient();
Sleep(1000);
}
catch (Exception& e)
{
cout << e.what() << endl;
}
}
}
};
int main()
{
HTTPserver h;
h.start();
return 0;
}
优点:
1.可以清晰的展示出错误原因,不像返回错误码那么模糊
2.许多第三方库使用异常,因此容易与这些结合使用
3.在测试框架里使用比较方便
缺点:
1.会打断执行流,函数有可能不在该返回的地方返回,这样使得代码的管理和调试困难
2.异常安全要使用RAII和不同编码实践。加大了代码量,需要大量的支持。
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。