人的一生会遇到很多大起大落,尤其是程序员.

  程序员写好的程序,论其消亡形式无非三种:无疾而终、自杀、他杀.

  当然作为一名程序员,最乐意看到自己写的程序能够无疾而终,因此尽快的学习异常处理机制是非常重要的!

  使自己的程序在遇到错误时能够克服错误,更健壮,而不是遇到错误就愤愤自杀.

  因此,在简述C++的异常机制之前,本文先来简述一下C语言中的异常处理机制.

  在C语言中,传统的错误处理方式有如下几种:

1.直接终止程序(自杀)

  例如:

int main(){
int a = 10;
int b = 20;
int c = a/0;
return 0;
}

  当用gcc编译完后,执行,会打印“浮点数例外”,然后程序结束.

  这种情况是不允许的,无条件终止程序的库无法运用到不能当机的程序里。

2.返回一个错误的值,附加错误码

  这种错误处理方式也很常见,比如我们用C语言打开一个文件失败时:

#include<stdio.h>
#include<errno.h>
int main(){
FILE* fp = fopen("test.txt","r");
if(NULL == fp){
printf("文件打开失败,错误码:%d\n",errno);
}
return 0;
}

  因为我当前处于Linux系统下,没有GetLastError()函数,所以在Linux下使用全局变量errno来演示.

  这种情况,比较常用,但是有时不合适,例如返回错误码是int,每个调用都要检查错误值,极不方便,也容易让程序规模加倍

3.返回一个合法的值,让程序处于某种非法的状态

  这种例子,最常见的就是atoi函数,例如:

#include<stdio.h>
#include<stdlib.h>
int main(){
int a = atoi("123456789");
int b = atoi("dasdasdcs");
printf("a = %d\n",a);
printf("b = %d\n",b);
return 0;
}

  虽然b为0,但是有一个全局变量表示了这个0属于非法值,不是由字符串0转过来的.

  这种情况,很容易误导调用者,万一调用者没有去检查全局变量errno或者通过其他方式检查错误,那是一个灾难,而且这种方式在并发的情况下不能很好工作

4.调用一个预先准备好在出现"错误"的情况下使用的函数.

  这种异常处理情况比较少见,那我就现场直编了一个:

#include<stdio.h>
#include<stdlib.h>
void DealError(){
printf("除数为0,老兄你在逗我?\n");
}
typedef void(*fun)();
int Div(int a,int b,fun callback){
if(b==0){
callback();
return 0;
}
return a/b;
} int main(){
printf("正常情况下的4/2 = %d\n",Div(4,2,DealError));
printf("调用错误处理函数的4/0 = %d\n",Div(4,0,DealError));
return 0;
}

5、通过暴力的方式解决

  暴力的方式有两种,abort()函数和常见exit()函数.例如依然处理除0问题,代码可以这样写:

#include<stdio.h>
#include<stdlib.h>
int Div(int a,int b){
if(b==0){
//exit(1); //直接退出
abort(); //在Windows下会弹出一个信息框
}
return a/b;
} int main(){
printf("正常情况下的4/2 = %d\n",Div(4,2));
printf("调用错误处理函数的4/0 = %d\n",Div(4,0));
return 0;
}

6、使用goto语句

  虽然,goto语句十分强大,但违背了程序的顺序执行,打乱的程序的执行流,盲目的使用goto语句可能会出现意想不到的错误,因此,并不推荐使用goto语句.

  但,尽管如此,为了探索C语言的异常处理机制,我还是实现一下goto语句处理异常的代码.

#include<stdio.h>
#include<stdlib.h>
int main(){
int a = 0;
int b = 0;
printf("请输入两个值:\n");
printf("a = ");
scanf("%d",&a);
printf("b = ");
scanf("%d",&b);
if(b==0){
goto Error;
}
printf("a/b = %d\n",a/b);
return 0;
Error:
printf("除数不能为0,程序异常退出!\n");
exit(-1);
}

7、使用setjmp()与longjmp()

  goto语句虽然可以跳来跳去,但标记与goto必须处于同一作用域内.

  想从一个函数跳转到另一个函数,就必须使用setjmp与longjmp组合.

  实例如下:

#include<stdio.h>
#include<setjmp.h> jmp_buf mark; int Div(int a,int b){
if(b==0){
longjmp(mark,1);  //会使state = 1
}
return a/b;
}
int main(){
int State = setjmp(mark); //保存寄存器相关信息,初始值为0
if(State==0){
Div(4,0);
}else{
switch(State){
case 1:
printf("除0异常!\n");
}
}
return 0;
}

注意事项:

  1、setjmp必须先调用,在异常位置通过调用longjmp以恢复先前被保存的程序执行点,否则将导致 不可预测的结果,甚至程序崩溃。

  2、在调用setjmp的函数返回之前调动longjmp,否则结果不可预料。

setjmp与longjmp存在以下缺陷:

  1、函数的使用者必须非常靠近函数调用的地方编写错误处理代码,无疑使代码变的臃肿笨拙。

  2、setjmp()和longjmp()并不能够很好的支持C++面向对象的语义。

  以上情况,便是C语言中的异常处理常见的机制,C++提供了更为完善的异常处理机制,我将在下文中简述.

【C++】异常简述(一):C语言中的异常处理机制的更多相关文章

  1. C语言中的异常处理机制

    #define try if(!setjmp(Jump_Buffer)) 返回try现场后重新执行判断,所以有两次执行. http://blog.csdn.net/tian_dao_chou_qin/ ...

  2. Java语言中的异常处理

    Java语言中的异常处理包括声明异常.抛出异常.捕获异常和处理异常四个环节.   throw用于抛出异常.   throws关键字可以在方法上声明该方法要抛出的异常,然后在方法内部通过throw抛出异 ...

  3. C语言中的异常处理

    一 前言: 异常处理,对于做面向对象开发的开发者来说是再熟悉不过了,例如在C#中有 try { ... } catch( Exception e){...} finally{ ..... } 在C++ ...

  4. .NET中的异常处理机制(一)

    1.异常处理的总体指导思想 学习C#中的异常处理机制,大概要了解以下几点: 首先,我们需要知道的事所有具体异常都是继承自System.Exception基类的. 其次,要熟悉FCL类库内置好的一些异常 ...

  5. 深入理解C++中的异常处理机制

    异常处理 增强错误恢复能力是提高代码健壮性的最有力的途径之一,C语言中采用的错误处理方法被认为是紧耦合的,函数的使用者必须在非常靠近函数调用的地方编 写错误处理代码,这样会使得其变得笨拙和难以使用.C ...

  6. Java中的异常处理机制《》

    异常机制已经成为判断一门编程语言是否成熟的标准,异常机制可以使程序中异常处理代码和正常业务代码分离,保证程序代码更加优雅,并提高程序健壮性. Java异常机制主要依赖于try.catch.finall ...

  7. C++中的异常处理机制

    C++中的捕获异常机制catch参数中实参的类型不同,采取的处理方式则不相同,且与普通的函数调用还不一样,具体表现为当抛出异常throw A()或throw obj时,对象会进行一次额外的对象复制操作 ...

  8. 16、java中的异常处理机制

    异常:就是程序在运行时出现不正常情况.异常由来:问题也是现实生活中一个具体的事物,也可以通过java的类的形式进行描述.并封装成对象. 其实就是java对不正常情况进行描述后的对象体现. 对于问题的划 ...

  9. Java中的异常处理机制的简单原理和应用

    异常是指java程序运行时(非编译)所发生的非正常情况或错误,与现实生活中的事件很相似,现实生活中的事件可以包含事件发生的时间.地点.人物.情节等信息,可以用一个对象来表示,Java使用面向对象的方式 ...

随机推荐

  1. mac下安装eclipse+CDT

    测试文件test.cpp #include <iostream>using namespace std; int main() {    cout << "!!!He ...

  2. 《JAVA与模式》之调停者模式

    调停者模式是对象的行为模式.调停者模式包装了一系列对象相互作用的方式,使得这些对象不必相互明显引用.从而使它们可以较松散地耦合.当这些对象中的某些对象之间的相互作用发生改变时,不会立即影响到其他的一些 ...

  3. Java Socket实战之二:多线程通信

    转自:http://developer.51cto.com/art/201202/317544.htm 上一篇文章说到怎样写一个最简单的Java Socket通信,但是在上一篇文章中的例子有一个问题就 ...

  4. mysqlnd cannot connect to MySQL 4.1+ using old authentication

    报这个错误主要是因为mysql使用了老的密码格式,而程序要求使用新的格式导致的,解决办法: SET old_passwords = 0; UPDATE mysql.user SET Password ...

  5. zoj3955:Saddle Point(想法题)

    传送门 题意 给出n*m的矩阵,询问所有子矩阵中鞍点的个数 鞍点定义:在行唯一最小,在列唯一最大 分析 我们遍历每个点,计算该点对于答案的贡献即可. 每个点的贡献为\((2^{numa[i][j]}) ...

  6. 洛谷P2082 区间覆盖(加强版)(珂朵莉树)

    传送门 虽然是黄题而且还是一波离散就能解决的东西 然而珂朵莉树还是很好用 相当于一开始区间全为0,然后每一次区间赋值,问最后总权值 珂朵莉树搞一搞就好了 //minamoto #include< ...

  7. DecimalFormat数字格式化用法“0”和“#”的区别

    1. 以“0”补位时: 如果数字少了,就会补“0”,小数和整数都会补: 如果数字多了,就切掉,但只切小数的末尾,整数不能切: 同时被切掉的小数位会进行四舍五入处理. 2. 以“#”补位时: 如果数字少 ...

  8. layui配置

    layui是一个全局变量,可以在任何地方访问到 layui.config 方法主配置信息(经测试好像不能添加额外属性) layui.setter读取主配置属性 layui.extend 方法增加主配置 ...

  9. mybatis 批量insert,update报错 The error occurred while setting parameters

    数据脚本执行正常,但是报错,搜索关键信息 The error occurred while setting parameters ,发现了解决帖子: http://blog.csdn.net/jing ...

  10. 洛谷 P3808 【模板】AC自动机(简单版)洛谷 P3796 【模板】AC自动机(加强版)

    https://www.cnblogs.com/gtarcoder/p/4820560.html 每个节点的后缀指针fail指针指向: 例如he,she,his,hers的例子(见蓝书P214): 7 ...