走进C++程序世界------异常处理
一、 概述
C++自身有着很强的纠错能力,发展到现在,已经建立了比較完好的异常处理机制。
C++的异常情况无非两种,一种是语法错误。即程序中出现了错误的语句,函数。结构和类,致使编译程序无法进行。还有一种是执行时发生的错误。一般与算法有关。
关于语法错误,不必多说,写代码时心细一点就能够解决。C++编译器的报错机制能够让我们轻松地解决这些错误。
第二种是执行时的错误,常见的有文件打开失败、数组下标溢出、系统内存不足等等。而一旦出现这些问题。引发算法失效、程序执行时无故停止等故障也是常有的。
这就要求我们在设计软件算法时要全面。比方针对文件打开失败的情况。保护的方法有非常多种。最简单的就是使用“return”命令,告诉上层调用者函数执行失败;第二种处理策略就是利用c++的异常机制,抛出异常。
二、c++异常处理机制
C++异常处理机制是一个用来有效地处理执行错误的很强大且灵活的工具,它提供了很多其它的弹性、安全性和稳固性,克服了传统方法所带来的问题.
异常的抛出和处理主要使用了下面三个keyword: try、 throw 、 catch 。
抛出异常即检測是否产生异常,在C++中。其採用throw语句来实现,假设检測到产生异常。则抛出异常。
该语句的格式为:
throw 表达式;
假设在try语句块的程序段中(包含在当中调用的函数)发现了异常,且抛弃了该异常,则这个异常就能够被try语句块后的某个catch语句所捕获并处理。捕获和处理的条件是被抛弃的异常的类型与catch语句的异常类型相匹配。因为C++使用数据类型来区分不同的异常,因此在推断异常时。throw语句中的表达式的值就没有实际意义,而表达式的类型就特别重要。
try-catch语句形式例如以下 :
- try
- {
- 包括可能抛出异常的语句。
- }
- catch(类型名 [形參名]) // 捕获特定类型的异常
- {
- }
- catch(类型名 [形參名]) // 捕获特定类型的异常
- {
- }
- catch(...) // 三个点则表示捕获全部类型的异常
- {
- }
【范例1】处理除数为0的异常。该范例将上述除数为0的异常能够用try/catch语句来捕获异常。并使用throw语句来抛出异常,从而实现异常处理。实现代码如代码清单1-1所看到的。
// 代码清单1-1
- #include<iostream.h> //包括头文件
- #include<stdlib.h>
- double fuc(double x, double y) //定义函数
- {
- if(y==0)
- {
- throw y; //除数为0。抛出异常
- }
- return x/y; //否则返回两个数的商
- }
- void main()
- {
- double res;
- try //定义异常
- {
- res=fuc(2,3);
- cout<<"The result of x/y is : "<<res<<endl;
- res=fuc(4,0); 出现异常,函数内部会抛出异常
- }
- catch(double) //捕获并处理异常
- {
- cerr<<"error of dividing zero.\n";
- exit(1); //异常退出程序
- }
- }
【范例2】自己定义异常类型 (在本文開始的代码中已经给出示范)
三、异常的接口声明
为了加强程序的可读性,使函数的用户可以方便地知道所使用的函数会抛出哪些异常,可以在函数的声明中列出这个函数可能抛出的全部异常类型。比如:
这表明函数fun()可能而且仅仅可能抛出类型(A,B,C,D)及其子类型的异常。
假设在函数的声明中没有包含异常的接口声明,则此函数能够抛出不论什么类型的异常,比如:
一个不会抛出不论什么类型异常的函数能够进行例如以下形式的声明:
五、异常处理中须要注意的问题
1. 假设抛出的异常一直没有函数捕获(catch),则会一直上传到c++执行系统那里。导致整个程序的终止
2. 一般在异常抛出后资源能够正常被释放。但注意假设在类的构造函数中抛出异常。系统是不会调用它的析构函数的。处理方法是:假设在构造函数中要抛出异常,则在抛出前要记得删除申请的资源。
3. 异常处理只通过类型而不是通过值来匹配的,所以catch块的參数能够没有參数名称,只须要參数类型。
4. 函数原型中的异常说明要与实现中的异常说明一致,否则easy引起异常冲突。
5. 应该在throw语句后写上异常对象时,throw先通过Copy构造函数构造一个新对象,再把该新对象传递给 catch.
那么当异常抛出后新对象怎样释放?
异常处理机制保证:异常抛出的新对象并不是创建在函数栈上,而是创建在专用的异常栈上,因此它才干够跨接多个函数而传递到上层,否则在栈清空的过程中就会被销毁。全部从try到throw语句之间构造起来的对象的析构函数将被自己主动调用。但假设一直上溯到main函数后还没有找到匹配的catch块,那么系统调用terminate()终止整个程序,这样的情况下不能保证全部局部对象会被正确地销毁。
6. catch块的參数推荐採用地址传递而不是值传递。不仅能够提高效率,还能够利用对象的多态性。
另外。派生类的异常扑获要放到父类异常扑获的前面,否则,派生类的异常无法被扑获。
7. 编写异常说明时,要确保派生类成员函数的异常说明和基类成员函数的异常说明一致,即派生类改写的虚函数的异常说明至少要和相应的基类虚函数的异常说明同样。甚至更加严格,更特殊。
#include "stdafx.h"
#include<stdlib.h>
#include<crtdbg.h>
#include <iostream>
// 内存泄露检測机制
#define _CRTDBG_MAP_ALLOC
#ifdef _DEBUG
#define new new(_NORMAL_BLOCK, __FILE__, __LINE__)
#endif // 自己定义异常类
class MyExcepction
{
public: // 构造函数,參数为错误代码
MyExcepction(int errorId)
{
// 输出构造函数被调用信息
std::cout << "MyExcepction is called" << std::endl;
m_errorId = errorId;
} // 拷贝构造函数
MyExcepction( MyExcepction& myExp)
{
// 输出拷贝构造函数被调用信息
std::cout << "copy construct is called" << std::endl;
this->m_errorId = myExp.m_errorId;
} ~MyExcepction()
{
// 输出析构函数被调用信息
std::cout << "~MyExcepction is called" << std::endl;
} // 获取错误码
int getErrorId()
{
return m_errorId;
} private:
// 错误码
int m_errorId;
}; int main(int argc, char* argv[])
{
// 内存泄露检測机制
_CrtSetDbgFlag( _CRTDBG_ALLOC_MEM_DF | _CRTDBG_LEAK_CHECK_DF ); // 能够改变错误码,以便抛出不同的异常进行測试
int throwErrorCode = 110; std::cout << " input test code :" << std::endl;
std::cin >> throwErrorCode; try
{
if ( throwErrorCode == 110)
{
MyExcepction myStru(110); // 抛出对象的地址 -> 由catch( MyExcepction* pMyExcepction) 捕获
// 这里该对象的地址抛出给catch语句,不会调用对象的拷贝构造函数
// 传地址是提倡的做法,不会频繁地调用该对象的构造函数或拷贝构造函数
// catch语句运行结束后,myStru会被析构掉
throw &myStru;
}
else if ( throwErrorCode == 119 )
{
MyExcepction myStru(119); // 抛出对象,这里会通过拷贝构造函数创建一个暂时的对象传出给catch
// 由catch( MyExcepction myExcepction) 捕获
// 在catch语句中会再次调用通过拷贝构造函数创建暂时对象复制这里传过去的对象
// throw结束后myStru会被析构掉
throw myStru;
}
else if ( throwErrorCode == 120 )
{
// 不提倡这种抛出方法
// 这样做的话,假设catch( MyExcepction* pMyExcepction)中不运行delete操作则会发生内存泄露 // 由catch( MyExcepction* pMyExcepction) 捕获
MyExcepction * pMyStru = new MyExcepction(120);
throw pMyStru;
}
else
{
// 直接创建新对象抛出
// 相当于创建了暂时的对象传递给了catch语句
// 由catch接收时通过拷贝构造函数再次创建暂时对象接收传递过去的对象
// throw结束后两次创建的暂时对象会被析构掉
throw MyExcepction(throwErrorCode);
}
}
catch( MyExcepction* pMyExcepction)
{
// 输出本语句被运行信息
std::cout << "运行了 catch( MyExcepction* pMyExcepction) " << std::endl; // 输出错误信息
std::cout << "error Code : " << pMyExcepction->getErrorId()<< std::endl; // 异常抛出的新对象并不是创建在函数栈上,而是创建在专用的异常栈上,不须要进行delete
//delete pMyExcepction;
}
catch ( MyExcepction myExcepction)
{
// 输出本语句被运行信息
std::cout << "运行了 catch ( MyExcepction myExcepction) " << std::endl; // 输出错误信息
std::cout << "error Code : " << myExcepction.getErrorId()<< std::endl;
}
catch(...)
{
// 输出本语句被运行信息
std::cout << "运行了 catch(...) " << std::endl; // 处理不了,又一次抛出给上级
throw ;
} // 暂停
int temp;
std::cin >> temp; return 0;
/*File : exception.cpp
*Auth : sjin
*Date : 20140515
*Mail : 413977243@qq.com
*/ #include <iostream>
#include <exception> using namespace std; class MyExcception:public exception{ public:
const char * what() const throw(){
return "this is a exception";
}
}; /*假设抛出异常,没有接收。程序将终止*/
void exception_1()throw(double,int,const char *,MyExcception)
{
int i;
cout << "输入1-3整数" <<endl;
cin >> i;
if(i == 1)throw MyExcception();
if(i == 2)throw "hello";
if(i== 3)throw 123; cout << "==========end =========" <<endl; }
int main()
{
try{ exception_1();
}catch(const char *ex){
cout <<" exceptiong is occur!" <<ex<< endl;
}catch(double e){
cout << e << endl;
}catch (int e){
cout << e <<endl;
}catch (MyExcception e){
cout << "MyExcception is occur!" << e.what()<< endl;
} return 0;
}
走进C++程序世界------异常处理的更多相关文章
- 成C++应用程序世界------异常处理
一. 概述 C++自身有着很强的纠错能力,发展到现在,已经建立了比較完好的异常处理机制.C++的异常情况无非两种,一种是语法错误,即程序中出现了错误的语句,函数,结构和类,致使编译程序无法进行.还有一 ...
- 走进C++程序世界-----继承和派生(2)
覆盖基类的函数 覆盖基类函数顾名思义就是在派生类中对基类的函数进行的重新定义.这里将会讲到下面的2个知识点: 1.隐藏基类的方法 2.调用基类的方法(隐式和显示调用基类的方法) /* *derive2 ...
- 走进C++程序世界------继承和派生
继承和派生 继承是面向对象编程语言的最重要方面之一,正确的使用继承可编写出设计良好,容易于维护和扩展的应用程序.下面是在其他博客中的总结: ****************************** ...
- 走进C++程序世界-----函数相关(全局变量)
全局变量 在函数外面定义的变量的作用域为全局,在程序的任何函数中都可用.与全局变量同名的局部变量不会修改全局变量的值,但会隐藏它.如果函数中有一个与全局变量同 名的局部变量时,则在函数中使用该名称时, ...
- 走进C++程序世界------IO标准库介绍
流概述 流是C++标准的组成部分,流的主要目标是,将从磁盘读取文件或将输入写入控制台屏幕的问题封装起来,创建流后程序猿就能够使用它.流将负责处理全部的细节. IO类库 在C++输入、输出操作是通 ...
- 走进C++程序世界-------类的定义和使用(数据成员和方法成员,析构函数,构造函数,内联实现)
类的成员简介 在C++中,可以通过声明一个类来穿件一种新的类型.类将一组变量(他们的类型通常不同)和一组相关的函数组合在一起.类可以有各种类型的变量组成,还可以包含其他类对象.成员变量称为数据成员它们 ...
- 走进C++程序世界-----operator new delete 重载
在C++ 的世界里,new 和delete 是keyword.而在C的世界里相相应的malloc和free是函数,关键C++的new和delete分析,详见前面的章节.这里就不在过多的介绍了.链接. ...
- 走进JavaWeb技术世界3:JDBC的进化与连接池技术
走进JavaWeb技术世界3:JDBC的进化与连接池技术 转载公众号[码农翻身] 网络访问 随着 Oracle, Sybase, SQL Server ,DB2, Mysql 等人陆陆续续住进数据库 ...
- [C#] 走进 LINQ 的世界
走进 LINQ 的世界 序 在此之前曾发表过三篇关于 LINQ 的随笔: 进阶:<LINQ 标准查询操作概述>(强烈推荐) 技巧:<Linq To Objects - 如何操作字符串 ...
随机推荐
- JSON与Javabean转换的几种形式
JSON格式的数据传递是最常用的方法之一,以下列出了常用的几种形态以及与Javabean之间的转换: String json1="{'name':'zhangsan','age':23,'i ...
- set集合玩法、三目运算
set是无序的,无法用下标获取值 创建set二种方式 1.第一种 s1=set() #创建一个空的set,看下面就知道为什么要这么创建一个空的集合 2.第二种 s2={11,22,33,44} # ...
- 几何:pick定理详解
一.概念 假设P的内部有I(P)个格点,边界上有B(P)个格点,则P的面积A(P)为:A(P)=I(P)+B(P)/2-1. 二.说明 Pick定理主要是计算格点多边形(定点全是格点的不自交图形)P的 ...
- luogu P2619 [国家集训队2]Tree I
题目链接 luogu P2619 [国家集训队2]Tree I 题解 普通思路就不说了二分增量,生成树check 说一下坑点 二分时,若黑白边权有相同,因为权值相同优先选白边,若在最有增量时出现黑白等 ...
- [BZOJ4237]稻草人(CDQ分治)
先按y排序,二分,两边递归下去,然后处理下半部分对上半部分的贡献,即左下点在下半部分,右上点在上半部分的合法矩形个数. 两个部分均按x排序,枚举右上点p,则左下点需要满足: 1.横坐标大于上半部分纵坐 ...
- 03-MyBatis主从实现代码读写分离应用以及实现
建立目录结构:
- Codeforces Round #348 (VK Cup 2016 Round 2, Div. 2 Edition) D. Little Artem and Dance 模拟
D. Little Artem and Dance 题目连接: http://www.codeforces.com/contest/669/problem/D Description Little A ...
- Codeforces Beta Round #9 (Div. 2 Only) B. Running Student 水题
B. Running Student 题目连接: http://www.codeforces.com/contest/9/problem/B Description And again a misfo ...
- Swify闭包
闭包:是字包含的匿名函数代码块,可以做为表达式.函数参数和函数返回值,闭包表达式的运算结果是一种函数类型.类似于 C# Lambda 表达式. 闭包表达式: {(参数列表)->返回类型 in 语 ...
- 典型案例收集-使用OpenVPN连通多个机房内网(转)(静态路由)
说明: 1.这篇文章主要是使用静态路由表实现的多个机房通过VPN连接后的子网机房互通. 2.OpenVPN使用的是桥接模式(server-bridge和dev tap),这个是关键点,只有这样设置才可 ...