C++中的异常语法

概述

C语言的异常缺陷在于返回值只有一个,可能出现二义性,没有统一的标准

  • C++中的异常必须有处理,如果没被处理,程序会调用terminate函数终止程序
  • 如果捕获到的异常不想处理,可以继续向上抛出 throw

异常的关键子

  • try 可能出现异常的地方写到 try中
  • throw 抛出异常
  • chtch 捕获异常
int Division(int a, int b)
{
	if(b == 0)
	{
		
		//抛出异常
		throw -1;
	}
}

void test()
{
			
	int a = 10;
	int b = 20;
	
	try
	{
		Division(a, b);
	}
	//捕获异常
	catch(int)
	{
		throw;  //不想处理可以继续向上抛出
		cout<<"异常捕获int"<<endl;
	}
	catch(...)
	{
		cout<<"其他的异常..."<<endl;
	}

}

自定义异常使用

class MyException
{	
public:
	void PrintError()
	{
		cout<<"自定义异常"<<endl;
	}	
}

int Division(int a, int b)
{
	if(b == 0)
	{
		
		//抛出异常
		throw MyException();  //匿名对象(当前行执行完成,立马被释放)
	}
}

void test()
{
			
	int a = 10;
	int b = 0;
	
	try
	{
		Division(a, b);
	}
	//捕获异常
	catch(MyException e)
	{
		e.PrintError();
	}
	catch(...)
	{
		cout<<"其他的异常..."<<endl;
	}

}

栈解旋

从try 开始起,到throw抛出异常前,栈上的数据都会被释放,释放的顺序与构造是相反的,整个过程称为栈解旋

#include <memory>

class Person
{
public:
	Person();
	~Person();
}


void test()
{

	//C++98的智能指针的使用 C11后废弃了
	auot_ptr<Person>p(new Person);
	
	//C11提供的的智能指针
	unique_ptr<Person>p2(new Person);

}

异常的接口声明

void func() throw(int)   //表示只能抛出 int类型的异常
{
	throw 1;
}


void func2() throw(intdouble)   //表示只能抛出 int和double类型的异常
{
	throw 1;
}

void func2() throw()   //表示不允许抛出异常
{
	throw 1;
}


异常变量的生命周期

class MyException
{	
public:

	MyException()
	{
		cout<<"构造函数"<<endl;
	}
	MyException(const MyException &e)
	{
		cout<<"拷贝构造"<<endl;
	}
	~MyException()
	{
		cout<<"析构函数"<<endl;
	}
}

void func()
{

	
	throw MyException();
}

void test()
{
	try
	{
		func();
	}
	catch(MyException &e)
	{
		cout<<" 自定义异常捕获"<<endl;
	}
}


C++标准异常库

异常名称描述
exception所有标准异常类的父类
bad_alloc当operator new and operator new[],请求分配内存失败时
bad_exception这是个特殊的异常,如果函数的异常抛出列表里声明了bad_exception异常,当函数内部抛出了异常抛出列表中没有的异常,这是调用的unexpected函数中若抛出异常,不论什么类型,都会被替换为bad_exception类型
bad_typeid使用typeid操作符,操作一个NULL指针,而该指针是带有虚函数的类,这时抛出bad_typeid异常
bad_cast使用dynamic_cast转换引用失败的时候
ios_base::failureio操作过程出现错误
logic_error逻辑错误,可以在运行前检测的错误
runtime_error运行时错误,仅在运行时才可以检测的错误

logic_error的子类:

异常名称描述
length_error试图生成一个超出该类型最大长度的对象时,例如vector的resize操作
domain_error参数的值域错误,主要用在数学函数中。例如使用一个负值调用只能操作非负数的函数
out_of_range超出有效范围
invalid_argument参数不合适。在标准库中,当利用string对象构造bitset时,而string中的字符不是’0’或’1’的时候,抛出该异常

runtime_error的子类:

异常名称描述
range_error计算结果超出了有意义的值域范围
overflow_erroroverflow_error 算术计算上溢
underflow_error算术计算下溢
invalid_argument参数不合适。在标准库中,当利用string对象构造bitset时,而string中的字符不是’0’或’1’的时候,抛出该异常

使用示例


#include<stdexcept>

class Person
{
public:
		Person(int age)
		{
			if(age < 0 || age > 150)
			{
			 	throw out_of_range("年龄越界!");
			}
			this->mAge = age;
		}
		
		int mAge;
}
void test()
{
	
	try
	{
		Person p(-10);
	}
	//catch(out_of_range &e)
	catch(exception &e)
	{
		cout<<e.what()<<endl;
	}


}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值