H面向对象异常处理YH课件

上传人:人*** 文档编号:567959182 上传时间:2024-07-22 格式:PPT 页数:52 大小:453KB
返回 下载 相关 举报
H面向对象异常处理YH课件_第1页
第1页 / 共52页
H面向对象异常处理YH课件_第2页
第2页 / 共52页
H面向对象异常处理YH课件_第3页
第3页 / 共52页
H面向对象异常处理YH课件_第4页
第4页 / 共52页
H面向对象异常处理YH课件_第5页
第5页 / 共52页
点击查看更多>>
资源描述

《H面向对象异常处理YH课件》由会员分享,可在线阅读,更多相关《H面向对象异常处理YH课件(52页珍藏版)》请在金锄头文库上搜索。

1、1-1知识回顾知识回顾I/OI/O流的概念流的概念输出流输出流输入流输入流异常处理异常处理第八章第八章理解异常处理的概念理解异常处理的概念掌握异常处理的实现掌握异常处理的实现掌握异常处理中对象的构造与析构掌握异常处理中对象的构造与析构理解名字空间的概述理解名字空间的概述本章目标本章目标异常处理是异常处理是C+C+语言中重要的错误处理机制,语言中重要的错误处理机制,是提高程序容错性的一种手段。异常处理主要是提高程序容错性的一种手段。异常处理主要针对程序运行时出现的各种异常情况,提供发针对程序运行时出现的各种异常情况,提供发现,捕获异常的手段,并尽量减少异常对程序现,捕获异常的手段,并尽量减少异常

2、对程序运行的影响。运行的影响。有的程序虽然经过编译、连接成为可以运行的程有的程序虽然经过编译、连接成为可以运行的程序,但在运行过程中难免会出现各种各样的问题,序,但在运行过程中难免会出现各种各样的问题,即使对那些所谓能即使对那些所谓能“正常运行正常运行”的程序而言也是的程序而言也是如此。如此。程序中潜在的异常问题程序中潜在的异常问题提出问题提出问题示例:示例:#include #include using namespace std;int main()float a,b,c;double x1,x2;cout请输入一元二次方程的系数请输入一元二次方程的系数a、b和和c的值的值:endl;co

3、uta;coutb;coutc;x1=(-b+sqrt(b*b-4*a*c)/(2*a);x2=(-b-sqrt(b*b-4*a*c)/(2*a);cout方程的实根是方程的实根是:x1=x1endl;cout方程的实根是方程的实根是:x2=x2endl;return 0;从上面例子可以看出:能够从上面例子可以看出:能够“正常运行正常运行”的程序可能的程序可能存在着许多潜在的存在着许多潜在的“隐患隐患”。程序运行可以检测到的。程序运行可以检测到的一些非正常情况称为异常(一些非正常情况称为异常(exception)。)。如除数为如除数为0、数组越界访问、内存空间不够、输入、数组越界访问、内存空间

4、不够、输入/输输出不正常(文件找不到、输入数据类型错等)等。出不正常(文件找不到、输入数据类型错等)等。异常是程序错误一种形式。异常是程序错误一种形式。分析问题分析问题程序中的错误按性质可分为语法错误、逻辑错误程序中的错误按性质可分为语法错误、逻辑错误和异常和异常3种。种。一般来说,异常的检测和处理要完成下列任务之一般来说,异常的检测和处理要完成下列任务之一:一:(1)让)让“用户用户”知道程序出现了异常,并退出知道程序出现了异常,并退出程序。程序。(2)让)让“用户用户”知道程序出现了异常,允许知道程序出现了异常,允许“用户用户”选择继续使用程序。选择继续使用程序。(3)在程序发生异常时,能

5、够在不打扰)在程序发生异常时,能够在不打扰“用户用户”的情况下继续程序的运行。的情况下继续程序的运行。C+C+语言异常处理机制的基本思想是将异常的检测与处理语言异常处理机制的基本思想是将异常的检测与处理分离。当在一个函数体中检测到异常条件存在,但无法确分离。当在一个函数体中检测到异常条件存在,但无法确定相应的处理方法时,将引发一个异常,并由函数直接或定相应的处理方法时,将引发一个异常,并由函数直接或间接调用检测并处理这个异常。间接调用检测并处理这个异常。这一基本思想用这一基本思想用3 3个保留字实现:个保留字实现:throwthrow、trytry和和catchcatch。在。在一般情况下,被

6、调用函数直接检测到异常条件的存在并使一般情况下,被调用函数直接检测到异常条件的存在并使用用throwthrow引发一个异常(注意,引发一个异常(注意,C+C+语言的异常是由程序员语言的异常是由程序员控制引发的,而不是由计算机硬件或程序运行环境控制的)控制引发的,而不是由计算机硬件或程序运行环境控制的);在上层调用函数中使用;在上层调用函数中使用trytry检测函数调用是否引发异常,检测函数调用是否引发异常,检测到的各种异常由检测到的各种异常由catchcatch捕获并作相应处理。捕获并作相应处理。异常处理实现异常处理实现在在VC+6.0环境中,为了使用异常处理机制,需要进行环境中,为了使用异常

7、处理机制,需要进行如下设置(默认设置):如下设置(默认设置):(1)选择菜单中的)选择菜单中的project。(2)在弹出的下拉菜单中选择)在弹出的下拉菜单中选择Setting命令,出现命令,出现Settings对话框。对话框。(3)打开)打开C/C+选项卡。选项卡。(4)在)在Category中选择中选择 C+ Language。(5)选中)选中Enable exception handling复选框。复选框。在在C+程序中,任何需要检测异常的语句(包程序中,任何需要检测异常的语句(包括函数调用)都必须在括函数调用)都必须在try语句块中执行,异语句块中执行,异常必须由紧跟着常必须由紧跟着t

8、ry语句后面的语句后面的catch语句来捕语句来捕获并处理。因而,获并处理。因而,try与与catch总是结合使用。总是结合使用。1、异常处理的语法、异常处理的语法throw、 try和和catch语句的一般语法如下:语句的一般语法如下:throw 表达式;表达式;try /try语句块语句块 catch(类型类型1 参数参数1) /针对类型针对类型1的异常处理的异常处理 catch(类型类型2 参数参数2) /针对类型针对类型1的异常处理的异常处理 .catch(类型(类型n 参数参数n) /针对类型针对类型1的异常处理的异常处理 异常处理的执行过程如下:异常处理的执行过程如下:(1 1)控

9、制通过正常的顺序执行到达)控制通过正常的顺序执行到达trytry语句,然后执行语句,然后执行trytry块内的块内的保护段。保护段。(2 2)如果在保护段执行期间没有引起异常,那么跟在)如果在保护段执行期间没有引起异常,那么跟在trytry块后的块后的catchcatch子句就不执行,程序从异常被抛掷的子句就不执行,程序从异常被抛掷的trytry块后跟随的最后一个块后跟随的最后一个catchcatch子句后面的语句继续执行下去。子句后面的语句继续执行下去。(3 3)如果在保护段执行期间或在保护段调用的任何函数中(直接)如果在保护段执行期间或在保护段调用的任何函数中(直接或间接的调用)有异常被抛

10、掷,则从通过或间接的调用)有异常被抛掷,则从通过throwthrow运算数创建的对象运算数创建的对象中创建一个异常对象(可能包含一个复制构造函数)。中创建一个异常对象(可能包含一个复制构造函数)。(4 4)如果匹配的处理器未找到,则运行函数)如果匹配的处理器未找到,则运行函数terminateterminate将被自动调将被自动调用,而函数用,而函数terminateterminate的默认功能是调用的默认功能是调用abortabort终止程序。终止程序。(5 5)如果找到了一个匹配的)如果找到了一个匹配的catchcatch处理程序,且它通过值进行捕获,处理程序,且它通过值进行捕获,则其形参

11、通过复制异常对象进行初始化。则其形参通过复制异常对象进行初始化。示例示例1:#include void main()char *buf;trybuf=new char512;if(buf=0)throw 内存分配错误内存分配错误!;cout内存分配成功内存分配成功!endl;catch(char *str)cout异常引发异常引发:strendl;示例示例2:#include int fun(int);void main()trycout4!=fun(4)endl;cout-2!=fun(-2)endl;cout5!=fun(5);catch (int n)coutn=n不能计算不能计算n!e

12、ndl;cout程序执行结束程序执行结束.endl;int fun(int n)if(n=0)throw n;int s=1;for(int i=1;i=n;i+)s*=i;return s;catch处理程序的出现次序很重要,因为在一个处理程序的出现次序很重要,因为在一个try块中,异常处理块中,异常处理程序是按照它出现的次序检查的。只要找到一个匹配的异常类型,程序是按照它出现的次序检查的。只要找到一个匹配的异常类型,后面的异常处理都将被忽略。后面的异常处理都将被忽略。示例示例3:#include void fun(int code)tryif(code=0) throw code; /引发

13、引发int类型的异常类型的异常if(code=1) throw x; /引发引发char类型的异常类型的异常if(code=2) throw 12.345; /引发引发double类型的类型的异常异常catch(int n)cout捕获整数类型:捕获整数类型:nendl;catch(char c)cout捕获字符类型:捕获字符类型:cendl;catch(double d)cout捕获双精度类型:捕获双精度类型:dendl;return;void main()fun(0);fun(1);fun(2);程序中同时列出多个程序中同时列出多个catch语句时,将以语句时,将以catch语句在程序中出

14、现语句在程序中出现的次序作类型匹配,并且只有一个匹配的的次序作类型匹配,并且只有一个匹配的catch语句被执行,其语句被执行,其他的他的catch语句将被忽略。语句将被忽略。catch(.)是一个特殊的捕获语句,可以捕获任何异常,因而在是一个特殊的捕获语句,可以捕获任何异常,因而在任何情况下其他任何情况下其他catch子句都不被检查。所以,子句都不被检查。所以,catch(.)应该放应该放在最后。在在最后。在VC+6.0中,若中,若catch(.)不是放在所有不是放在所有catch(.)语句的最后,则会出现编译错误。语句的最后,则会出现编译错误。2、捕获异常、捕获异常示例示例1:#includ

15、e void fun(int code)tryif(code=0) throw code; /引发引发int类型的异常类型的异常 if(code=1) throw x; /引发引发char类型的异常类型的异常 if(code=2) throw 12.345; /引发引发double类型的异常类型的异常catch(int n) cout捕获整数类型捕获整数类型.nendl;catch(.) cout默认捕获默认捕获.endl;return;void main()fun(0);fun(1);fun(2);从前面异常处理的例子可以看出,调用一个函数时,除从前面异常处理的例子可以看出,调用一个函数时,

16、除了了解函数的参数与返回值外,还必须了解函数的异常了了解函数的参数与返回值外,还必须了解函数的异常引发方式,以便设计异常处理程序,应付函数调用过程引发方式,以便设计异常处理程序,应付函数调用过程中引发的异常。中引发的异常。异常的引发与捕获已成为函数之间界面的一部分,有必异常的引发与捕获已成为函数之间界面的一部分,有必要在函数原型中也列出异常引发。要在函数原型中也列出异常引发。例如:例如:void fun(int i) throw(t1,t2,t3);3、带有异常声明的函数原型、带有异常声明的函数原型C+的异常处理机制不仅能够处理各种不同类型的异常,的异常处理机制不仅能够处理各种不同类型的异常,

17、还具有为抛出异常前构造所有局部对象自动调用析构函数还具有为抛出异常前构造所有局部对象自动调用析构函数的能力。的能力。在程序中,找到一个匹配的在程序中,找到一个匹配的catch异常处理后,如果异常处理后,如果catch子句的异常类型说明是一个值参数,则其初始化方式是复子句的异常类型说明是一个值参数,则其初始化方式是复制被抛出的异常对象。如果制被抛出的异常对象。如果catch子句的异常类型说明是子句的异常类型说明是一个引用,则其初始化方式是使该引用指向异常对象。一个引用,则其初始化方式是使该引用指向异常对象。当当catch子句的异常类型说明参数被初始化后,便于始展子句的异常类型说明参数被初始化后,

18、便于始展开栈的过程。这包括将从对应的开栈的过程。这包括将从对应的try块开始到异常被抛出之块开始到异常被抛出之间构造(且尚未析构)的所有自动对象进行析构。析构的间构造(且尚未析构)的所有自动对象进行析构。析构的次序与构造的次序相反。然后程序从最后一个次序与构造的次序相反。然后程序从最后一个catch处理处理之后开始恢复执行。之后开始恢复执行。异常处理中对象的构造与析构异常处理中对象的构造与析构示例:示例:#include void fun(void);class Apublic:A() ;A() ;const char *ShowReason() const /异常处理成员函数异常处理成员函数

19、return 异常在异常在A类中类中;class Bpublic:B();B();B:B()coutB构造函数构造函数endl;B:B()coutB析构函数析构函数endl; void fun()B b;coutfun():抛掷一个抛掷一个A异常异常endl;throw A();void main()cout进入进入main()endl;trycout在在try块中调用块中调用fun()endl;fun();catch( A E)cout在在catch处理器捕获一个异常类型:处理器捕获一个异常类型:;coutE.ShowReason()endl;catch(char *str)cout捕获其他

20、异常:捕获其他异常:strendl;cout返回返回main()endl;在名字空间中可以放入这样的声明:类、变量(以及它在名字空间中可以放入这样的声明:类、变量(以及它们的初始化)、函数(以及它们的定义)、模板以及其们的初始化)、函数(以及它们的定义)、模板以及其他名字空间。从而这些变量或函数都与该名字空间相关他名字空间。从而这些变量或函数都与该名字空间相关联。联。名字空间概述名字空间概述1、名字空间的定义、名字空间的定义保留字(保留字(namespace)用于定义名字空间。名字空间必)用于定义名字空间。名字空间必须在程序的全局作用域内定义,不能在函数内或类内部须在程序的全局作用域内定义,不

21、能在函数内或类内部定义,最外层名字空间的名字必须在程序的全局作用域定义,最外层名字空间的名字必须在程序的全局作用域惟一。惟一。名字空间可以分多次定义,即可以先在初始定义中定义名字空间可以分多次定义,即可以先在初始定义中定义一部分成员,然后在扩展定义中再定义另一部分成员,一部分成员,然后在扩展定义中再定义另一部分成员,或者再定义初始时声明函数原型。初始定义和扩展定义或者再定义初始时声明函数原型。初始定义和扩展定义的语法格式相同。的语法格式相同。保留字保留字using用于声明程序要引入的名字空间成员,或都用于声明程序要引入的名字空间成员,或都用于指示程序要引用的名字空间。在声明引用名字空间用于指示

22、程序要引用的名字空间。在声明引用名字空间的某个成员之前,成员必须已经在名字空间中进行了声的某个成员之前,成员必须已经在名字空间中进行了声明或进行了定义。明或进行了定义。示例示例#include namespace NS1 /初始定义名字空间初始定义名字空间NS1extern int x; /说明整型变量说明整型变量xvoid fun(int); /说明函数说明函数fun(int)void fun(long) /定义函数定义函数fun(long)coutProcessing a long argument endl;namespace NS1 /扩展定义名字空间扩展定义名字空间NS1int x=

23、5; /定义整形变量定义整形变量xvoid fun(int) /定义函数定义函数fun(int)coutProcessing a int argumentendl;void main()int y=20;using NS1:x; /说明引用变量说明引用变量xusing:NS1:fun; /说明引用函数说明引用函数fun()x=10;fun(4);fun(4L);coutx=xendl;couty=yendl;1、访问名字空间的成员、访问名字空间的成员访问名字空间的成员有访问名字空间的成员有4种方式:种方式:(1)直接访问成员)直接访问成员格式如下:格式如下:名字空间名字名字空间名字:成员名字成

24、员名字因此,直接访问总能惟一地访问指定名字空间的成员。因此,直接访问总能惟一地访问指定名字空间的成员。(2)指定名字空间(使用)指定名字空间(使用using namespace 语句)语句)指定名字空间的格式如下:指定名字空间的格式如下:using namespace 名字空间;名字空间;/直接使用成员名字直接使用成员名字(3)声明引用成员(使用)声明引用成员(使用using语句)语句)声明引用成员的格式如下:声明引用成员的格式如下:using 名字空间名字空间:名字名字/直接使用成员名字直接使用成员名字(4)使用别名法)使用别名法声明引用成员的格式如下:声明引用成员的格式如下:namespa

25、ce 别名别名=名字空间;名字空间;/使用使用“别名别名:成员名字成员名字”本方法与直接方法成员方法类似,只是加了一个别名。例如有以下两本方法与直接方法成员方法类似,只是加了一个别名。例如有以下两个名字空间个名字空间NSA和和NSB,分别声明了同名的类模板:,分别声明了同名的类模板:namespace NSA template class Array private: T *ia; int ssize; ;以上的类以上的类Array被封装在名字空间被封装在名字空间NSA中,在使该类前,中,在使该类前,必须使必须使NSA名字空间可见。名字空间可见。namespace NSB template c

26、lass Array private: T *ia; int ssize; ;以上的类以上的类Array被封装在名字空间被封装在名字空间NSB中,在使该类前,必须使中,在使该类前,必须使NSB名字空间可见。因为使用了名字空间,两个名字空间可见。因为使用了名字空间,两个Array类分别在不同的名类分别在不同的名字空间中,所以不会存在冲突。其中四种使用方式如下:字空间中,所以不会存在冲突。其中四种使用方式如下:第一种用法:直接用法。第一种用法:直接用法。NSA:Array a;NSB:Array b;第二种用法:指定名字空间。第二种用法:指定名字空间。using namespace NSA;Arr

27、ay a;using namespace NSB;Array b;第三种用法:声明引用成员。第三种用法:声明引用成员。using NSA:Array;Array a;using NSB:Array; /错误,存在同名的成员错误,存在同名的成员Array b;值得注意的是,上述语句是错误的,应将两个类模板改为不同的名字,值得注意的是,上述语句是错误的,应将两个类模板改为不同的名字,如将如将NSB中的中的Array改为改为Array1,则以下语句是正确的:则以下语句是正确的:using NSA:Array;Array a;using NSB:Array1;Array b;第四种用法:使用别名法。第

28、四种用法:使用别名法。namespace us=NSA;namespace ms=NSB;us:Array a;ms:Array b;示例:示例:#include namespace NS1int x=10; namespace NS2int x=20; void main()using NS1:x;coutx=xendl;using NS2:x;coutx=xendl;2、使用作用域运算符、使用作用域运算符“:”访问成员访问成员当名字空间的成员和程序的全局标识符同名时,可以通当名字空间的成员和程序的全局标识符同名时,可以通过作用域运算符过作用域运算符“:”既定程序的全局标识符;当名字既定程序

29、的全局标识符;当名字空间的成员和程序的标识符同名时,首先访问的是程序空间的成员和程序的标识符同名时,首先访问的是程序的局部标识符。的局部标识符。示例:示例:#include int x=20; /全局变量全局变量namespace NS1int x=10;void main()using namespace NS1;coutx=:xendl;3、名字空间的嵌套、名字空间的嵌套名字空间也可以像类那样嵌套,形成多个层次的作用域,名字空间也可以像类那样嵌套,形成多个层次的作用域,因此,在访问名字空间的成员时,就有可能使用多个域因此,在访问名字空间的成员时,就有可能使用多个域运算符。运算符。示例:示例

30、:#include namespace NS1 /NS1的初始定义的初始定义int x=10;void fun1()coutNS1s fun1()endl;namespace NS2int y=20;void fun2()coutNS2s fun2()endl;using NS1:fun1; /using说明,全局名字空间限定说明,全局名字空间限定fun1using NS1:x; /using说明,全局名字空间限定说明,全局名字空间限定xusing NS1:NS2:fun2; /using说明,多重名字空间限定说明,多重名字空间限定fun2using NS1:NS2:y; /using说明,多

31、重名字空间限定说明,多重名字空间限定yvoid main()fun1();fun2();coutx=xendl;couty=yendl;4、 std名字空间名字空间本章前面的程序都是使用标准本章前面的程序都是使用标准C+编写的,其头文件都编写的,其头文件都带有带有.h扩展名,而扩展名,而ANSI/ISO标准标准C+头文件不带扩展头文件不带扩展名。这是因为名。这是因为C+是从是从C发展而来,某些头文件,如发展而来,某些头文件,如math.h都是从都是从C中沿袭过来的,而诸如中沿袭过来的,而诸如iostream.h、iostreamip.h和和fstream.h等是特别为等是特别为C+设计的。当设

32、计的。当一个头文件被包含进某一个程序中,头文件中的全局标一个头文件被包含进某一个程序中,头文件中的全局标识符也就变成了程序中的全局标识符。在识符也就变成了程序中的全局标识符。在ANSI/ISO标标准准c+中,为了利用名字空间机制的先进之处,所有的中,为了利用名字空间机制的先进之处,所有的头文件都被修改过,这样的标识符都被声明在头文件都被修改过,这样的标识符都被声明在std名字空名字空间中。间中。Std是是C+标准库的名字空间,标准库的名字空间,C+标准库中的标准库中的所有定义都被定义在这个名字空间中。所有定义都被定义在这个名字空间中。标准标准C+和和ANSI/ISO标准标准C+中头文件的对应关

33、系中头文件的对应关系标准标准C+头文件名头文件名ANSI/ISO标准头文件名标准头文件名assert.hcassertctype.hcctypefloat.hcfloatfstream.hfstreamiostream.h.iostreamiomanip.hiomanipimits.hclimitsmath.hcmathStdlib.hcstdlibString.hcstring1、直接指定标识符、直接指定标识符在所有特定的标识符剪辑上在所有特定的标识符剪辑上“std:”前缀。前缀。#include /使用使用C+标准库标准库void main()int n=100;std:coutstd:h

34、exnstd:endl;2、使用、使用using关键字关键字对于每个特定的标识符,都用对于每个特定的标识符,都用“using std: 标识符标识符”进行声明。进行声明。#include /使用使用C+标准库标准库using std:cout;using std:endl;using std:hex;void main()int n=100;couthexnendl;3、使用、使用using namespace std这是一种最方便的方法,从而使这是一种最方便的方法,从而使std命名空间的内定义的命名空间的内定义的所有标识符都有效,就好像它们被声明为全局变量一样。所有标识符都有效,就好像它们被声明为全局变量一样。#include /使用使用C+标准库标准库using namespace std;void main()int n=100;couthexnendl;异常和异常处理的概念异常和异常处理的概念异常处理的实现异常处理的实现异常处理中对象的构造与析构异常处理中对象的构造与析构名字空间的概述名字空间的概述本章总结本章总结

展开阅读全文
相关资源
正为您匹配相似的精品文档
相关搜索

最新文档


当前位置:首页 > 办公文档 > 工作计划

电脑版 |金锄头文库版权所有
经营许可证:蜀ICP备13022795号 | 川公网安备 51140202000112号