谭浩强《C程序设计》课件第14章课件

上传人:阳*** 文档编号:100731088 上传时间:2022-06-03 格式:PPT 页数:62 大小:191KB
收藏 版权申诉 举报 下载
谭浩强《C程序设计》课件第14章课件_第1页
第1页 / 共62页
谭浩强《C程序设计》课件第14章课件_第2页
第2页 / 共62页
谭浩强《C程序设计》课件第14章课件_第3页
第3页 / 共62页
资源描述:

《谭浩强《C程序设计》课件第14章课件》由会员分享,可在线阅读,更多相关《谭浩强《C程序设计》课件第14章课件(62页珍藏版)》请在装配图网上搜索。

1、第第14章章 C+工具工具14.1 异常处理异常处理14.2 命名空间命名空间14.3 使用早期的函数库使用早期的函数库在在C+发展的后期,有时发展的后期,有时C+编译系统根据实际工作编译系统根据实际工作的需要,增加了一些功能,作为工具来使用,其中的需要,增加了一些功能,作为工具来使用,其中主要有模板主要有模板(包括函数模板和类模板包括函数模板和类模板)、异常处理、命、异常处理、命名空间和运行时类型识别,以帮助程序设计人员更名空间和运行时类型识别,以帮助程序设计人员更方便地进行程序的设计和调试工作。方便地进行程序的设计和调试工作。1997年年ANSI C+委员会将它们纳入了委员会将它们纳入了A

2、NSI C+标准,建议所有标准,建议所有的的C+编译系统都能实现这些功能。这些工具是非常编译系统都能实现这些功能。这些工具是非常有用的,有用的,C+的使用者应当尽量使用这些工具。的使用者应当尽量使用这些工具。程序编制者不仅要考虑程序没有错误的理想情况,程序编制者不仅要考虑程序没有错误的理想情况,更要考虑程序存在错误时的情况,应该能够尽快地更要考虑程序存在错误时的情况,应该能够尽快地发现错误,消除错误。发现错误,消除错误。程序中常见的错误有两大类程序中常见的错误有两大类: 语法错误和运行错误。语法错误和运行错误。在编译时,编译系统能发现程序中的语法错误。在编译时,编译系统能发现程序中的语法错误。

3、有的程序虽然能通过编译,也能投入运行。但是在有的程序虽然能通过编译,也能投入运行。但是在运行过程中会出现异常,得不到正确的运行结果,运行过程中会出现异常,得不到正确的运行结果,甚至导致程序不正常终止,或出现死机现象。这类甚至导致程序不正常终止,或出现死机现象。这类错误比较隐蔽,不易被发现,往往耗费许多时间和错误比较隐蔽,不易被发现,往往耗费许多时间和精力。这成为程序调试中的一个难点。精力。这成为程序调试中的一个难点。14.1 异常处理异常处理 14.1.1 异常处理的任务异常处理的任务在设计程序时,应当事先分析程序运行时可能出现在设计程序时,应当事先分析程序运行时可能出现的各种意外的情况,并且

4、分别制订出相应的处理方的各种意外的情况,并且分别制订出相应的处理方法,这就是程序的异常处理的任务。法,这就是程序的异常处理的任务。在运行没有异常处理的程序时,如果运行情况出现在运行没有异常处理的程序时,如果运行情况出现异常,由于程序本身不能处理,程序只能终止运行。异常,由于程序本身不能处理,程序只能终止运行。如果在程序中设置了异常处理机制,则在运行情况如果在程序中设置了异常处理机制,则在运行情况出现异常时,由于程序本身已规定了处理的方法,出现异常时,由于程序本身已规定了处理的方法,于是程序的流程就转到异常处理代码段处理。用户于是程序的流程就转到异常处理代码段处理。用户可以指定进行任何的处理。可

5、以指定进行任何的处理。需要说明,只要出现与人们期望的情况不同,都可需要说明,只要出现与人们期望的情况不同,都可以认为是异常,并对它进行异常处理。因此,所谓以认为是异常,并对它进行异常处理。因此,所谓异常处理指的是对运行时出现的差错以及其他例外异常处理指的是对运行时出现的差错以及其他例外情况的处理。情况的处理。在一个小的程序中,可以用比较简单的方法处理异在一个小的程序中,可以用比较简单的方法处理异常。但是在一个大的系统中,如果在每一个函数中常。但是在一个大的系统中,如果在每一个函数中都设置处理异常的程序段,会使程序过于复杂和庞都设置处理异常的程序段,会使程序过于复杂和庞大。因此,大。因此,C+采

6、取的办法是采取的办法是: 如果在执行一个函数如果在执行一个函数过程中出现异常,可以不在本函数中立即处理,而过程中出现异常,可以不在本函数中立即处理,而是发出一个信息,传给它的上一级是发出一个信息,传给它的上一级(即调用它的函数即调用它的函数),它的上级捕捉到这个信息后进行处理。如果上一级它的上级捕捉到这个信息后进行处理。如果上一级的函数也不能处理,就再传给其上一级,由其上一的函数也不能处理,就再传给其上一级,由其上一级处理。如此逐级上送,如果到最高一级还无法处级处理。如此逐级上送,如果到最高一级还无法处理,最后只好异常终止程序的执行。理,最后只好异常终止程序的执行。14.1.2 异常处理的方法

7、异常处理的方法这样做使异常的发现与处理不由同一函数来完成。这样做使异常的发现与处理不由同一函数来完成。好处是使底层的函数专门用于解决实际任务,而不好处是使底层的函数专门用于解决实际任务,而不必再承担处理异常的任务,以减轻底层函数的负担,必再承担处理异常的任务,以减轻底层函数的负担,而把处理异常的任务上移到某一层去处理。这样可而把处理异常的任务上移到某一层去处理。这样可以提高效率。以提高效率。C+处理异常的机制是由处理异常的机制是由3个部分组成的,即检查个部分组成的,即检查(try)、抛出抛出(throw)和捕捉和捕捉(catch)。把需要检查的语把需要检查的语句放在句放在try块中,块中,th

8、row用来当出现异常时发出一个异用来当出现异常时发出一个异常信息,而常信息,而catch则用来捕捉异常信息,如果捕捉到则用来捕捉异常信息,如果捕捉到了异常信息,就处理它。了异常信息,就处理它。例例14.1 给出三角形的三边给出三角形的三边a,b,c,求三角形的面积。求三角形的面积。只有只有a+bc,b+ca,c+ab时才能构成三角形。设置异时才能构成三角形。设置异常处理,对不符合三角形条件的输出警告信息,不常处理,对不符合三角形条件的输出警告信息,不予计算。予计算。先写出没有异常处理时的程序先写出没有异常处理时的程序:#include #include using namespace std;

9、int main( )double triangle(double,double,double); double a,b,c; cinabc; while(a0 & b0 & c0) couttriangle(a,b,c)abc; return 0;double triangle(double a,double b,double c)double area; double s=(a+b+c)/2; area=sqrt(s*(s-a)*(s-b)*(s-c); return area;运行情况如下运行情况如下:6 5 4 (输入输入a,b,c的值的值) 9.92157 (输出三角形的面积输出三角

10、形的面积)1 1.5 2 (输入输入a,b,c的值的值)0.726184 (输出三角形的面积输出三角形的面积)1 2 1 (输入输入a,b,c的值的值)0 (输出三角形的面积,此结果显然不对输出三角形的面积,此结果显然不对,因为不是三角形因为不是三角形)1 0 6 (输入输入a,b,c的值的值) (结束结束)修改程序,在函数修改程序,在函数traingle中对三角形条件进行检查,中对三角形条件进行检查,如果不符合三角形条件,就抛出一个异常信息,在如果不符合三角形条件,就抛出一个异常信息,在主函数中的主函数中的try-catch块中调用块中调用traingle函数,检测有函数,检测有无异常信息,

11、并作相应处理。修改后的程序如下无异常信息,并作相应处理。修改后的程序如下:#include #include using namespace std;void main( )double triangle(double,double,double); double a,b,c; cinabc; try/在在try块中包含要检查的函数块中包含要检查的函数 while(a0 & b0 & c0) couttriangle(a,b,c)abc; catch(double) /用用catch捕捉异常信息并作相应处理捕捉异常信息并作相应处理 couta=a,b=b,c=c,that is not a t

12、riangle!endl; coutendendl;double triangle(double a,double b,double c) /计算三角形的面积的函数计算三角形的面积的函数double s=(a+b+c)/2; if (a+b=c|b+c=a|c+a=b) throw a; /当不符合三角形条件抛出异常信息当不符合三角形条件抛出异常信息 return sqrt(s*(s-a)*(s-b)*(s-c);程序运行结果如下程序运行结果如下:6 5 4 (输入输入a,b,c的值的值) 9.92157 (计算出三角形的面积计算出三角形的面积)1 1.5 2 (输入输入a,b,c的值的值)0

13、.726184 (计算出三角形的面积计算出三角形的面积)1 2 1 (输入输入a,b,c的值的值)a=1,b=2,c=1, that is not a triangle! (异常处理异常处理)end现在结合程序分析怎样进行异常处理。现在结合程序分析怎样进行异常处理。(1) 首先把可能出现异常的、需要检查的语句或程序首先把可能出现异常的、需要检查的语句或程序段放在段放在try后面的花括号中。后面的花括号中。(2) 程序开始运行后,按正常的顺序执行到程序开始运行后,按正常的顺序执行到try块,开块,开始执行始执行try块中花括号内的语句。如果在执行块中花括号内的语句。如果在执行try块内块内的语句

14、过程中没有发生异常,则的语句过程中没有发生异常,则catch子句不起作用,子句不起作用,流程转到流程转到catch子句后面的语句继续执行。子句后面的语句继续执行。(3) 如果在执行如果在执行try块内的语句块内的语句(包括其所调用的函数包括其所调用的函数)过程中发生异常,则过程中发生异常,则throw运算符抛出一个异常信息。运算符抛出一个异常信息。throw抛出异常信息后,流程立即离开本函数,转到抛出异常信息后,流程立即离开本函数,转到其上一级的函数其上一级的函数(main 函数函数)。throw抛出什么样的数据由程序设计者自定,可以是抛出什么样的数据由程序设计者自定,可以是任何类型的数据。任

15、何类型的数据。(4) 这个异常信息提供给这个异常信息提供给try-catch结构,系统会寻找结构,系统会寻找与之匹配的与之匹配的catch子句。子句。(5) 在进行异常处理后,程序并不会自动终止,继续在进行异常处理后,程序并不会自动终止,继续执行执行catch子句后面的语句。子句后面的语句。由于由于catch子句是用来处理异常信息的,往往被称为子句是用来处理异常信息的,往往被称为catch异常处理块或异常处理块或catch异常处理器。异常处理器。下面讲述异常处理的语法。下面讲述异常处理的语法。throw语句一般是由语句一般是由throw运算符和一个数据组成的,运算符和一个数据组成的,其形式为其

16、形式为throw 表达式表达式;try-catch的结构为的结构为try 被检查的语句被检查的语句 catch(异常信息类型异常信息类型 变量名变量名) 进行异常处理的语句进行异常处理的语句说明说明:(1) 被检测的函数必须放在被检测的函数必须放在try块中,否则不起作用。块中,否则不起作用。(2) try块和块和catch块作为一个整体出现,块作为一个整体出现,catch块是块是try-catch结构中的一部分,必须紧跟在结构中的一部分,必须紧跟在try块之后,不能块之后,不能单独使用,在二者之间也不能插入其他语句。但是单独使用,在二者之间也不能插入其他语句。但是在一个在一个try-catc

17、h结构中,可以只有结构中,可以只有try块而无块而无catch块。块。即在本函数中只检查而不处理,把即在本函数中只检查而不处理,把catch处理块放在处理块放在其他函数中。其他函数中。(3) try和和catch块中必须有用花括号括起来的复合语块中必须有用花括号括起来的复合语句,即使花括号内只有一个语句,也不能省略花括句,即使花括号内只有一个语句,也不能省略花括号。号。(4) 一个一个try-catch结构中只能有一个结构中只能有一个try块,但却可以块,但却可以有多个有多个catch块,以便与不同的异常信息匹配。块,以便与不同的异常信息匹配。(5) catch后面的圆括号中,一般只写异常信息

18、的类型后面的圆括号中,一般只写异常信息的类型名,名, 如如catch(double)catch只检查所捕获异常信息的类型,而不检查它们只检查所捕获异常信息的类型,而不检查它们的值。因此如果需要检测多个不同的异常信息,应的值。因此如果需要检测多个不同的异常信息,应当由当由throw抛出不同类型的异常信息。抛出不同类型的异常信息。异常信息可以是异常信息可以是C+系统预定义的标准类型,也可以系统预定义的标准类型,也可以是用户自定义的类型是用户自定义的类型(如结构体或类如结构体或类)。如果由。如果由throw抛出的信息属于该类型或其子类型,则抛出的信息属于该类型或其子类型,则catch与与throw二

19、者匹配,二者匹配,catch捕获该异常信息。捕获该异常信息。catch还可以有另外一种写法,即除了指定类型名外,还可以有另外一种写法,即除了指定类型名外,还指定变量名,如还指定变量名,如catch(double d)此时如果此时如果throw抛出的异常信息是抛出的异常信息是double型的变量型的变量a,则则catch在捕获异常信息在捕获异常信息a的同时,还使的同时,还使d获得获得a的值,的值,或者说或者说d得到得到a的一个拷贝。什么时候需要这样做呢?的一个拷贝。什么时候需要这样做呢?有时希望在捕获异常信息时,还能利用有时希望在捕获异常信息时,还能利用throw抛出的抛出的值,如值,如catc

20、h(double d) coutthrow d;这时会输出这时会输出d的值的值(也就是也就是a值值)。当抛出的是类对象时,。当抛出的是类对象时,有时希望在有时希望在catch块中显示该对象中的某些信息。这块中显示该对象中的某些信息。这时就需要在时就需要在catch的参数中写出变量名的参数中写出变量名(类对象名类对象名)。(6) 如果在如果在catch子句中没有指定异常信息的类型,而子句中没有指定异常信息的类型,而用了删节号用了删节号“”,则表示它可以捕捉任何类型的,则表示它可以捕捉任何类型的异常信息,如异常信息,如catch() coutOKendl;它能捕捉所有类型的异常信息,并输出它能捕捉

21、所有类型的异常信息,并输出OK。这种这种catch子句应放在子句应放在trycatch结构中的最后,相当结构中的最后,相当于于“其他其他”。如果把它作为第一个。如果把它作为第一个catch子句,则后子句,则后面的面的catch子句都不起作用。子句都不起作用。(7) trycatch结构可以与结构可以与throw出现在同一个函数中,出现在同一个函数中,也可以不在同一函数中。当也可以不在同一函数中。当throw抛出异常信息后,抛出异常信息后,首先在本函数中寻找与之匹配的首先在本函数中寻找与之匹配的catch,如果在本函如果在本函数中无数中无trycatch结构或找不到与之匹配的结构或找不到与之匹配

22、的catch,就就转到离开出现异常最近的转到离开出现异常最近的trycatch结构去处理。结构去处理。(8) 在某些情况下,在在某些情况下,在throw语句中可以不包括表达式,语句中可以不包括表达式,如如throw;表示表示“我不处理这个异常,请上级处理我不处理这个异常,请上级处理”。(9) 如果如果throw抛出的异常信息找不到与之匹配的抛出的异常信息找不到与之匹配的catch块,那么系统就会调用一个系统函数块,那么系统就会调用一个系统函数terminate,使程序终止运行。使程序终止运行。例例14.2 在函数嵌套的情况下检测异常处理。在函数嵌套的情况下检测异常处理。这是一个简单的例子,用来

23、说明在这是一个简单的例子,用来说明在try块中有函数嵌块中有函数嵌套调用的情况下抛出异常和捕捉异常的情况。请自套调用的情况下抛出异常和捕捉异常的情况。请自己先分析以下程序。己先分析以下程序。#include using namespace std;int main( )void f1( ); try f1( );/调用调用f1( ) catch(double) coutOK0!endl; coutend0endl; return 0;void f1( )void f2( ); try f2( ); /调用调用f2( ) catch(char) coutOK1!; coutend1endl;vo

24、id f2( )void f3( ); try f3( ); /调用调用f3( ) catch(int) coutOk2!endl; coutend2endl;void f3( )double a=0; try throw a; /抛出抛出double类型异常信息类型异常信息 catch(float) coutOK3!endl; coutend3endl;分分3种情况分析运行情况种情况分析运行情况: (1) 执行上面的程序。图执行上面的程序。图14.1为有函数嵌套时异常为有函数嵌套时异常处理示意图。处理示意图。图图14.1程序运行结果如下程序运行结果如下: OK0!(在主函数中捕获异常在主函数

25、中捕获异常)end0 (执行主函数中最后一个语句时的输出执行主函数中最后一个语句时的输出)(2) 如果将如果将f3函数中的函数中的catch子句改为子句改为catch(double),而程序中其他部分不变,则程序运行结果如下而程序中其他部分不变,则程序运行结果如下:OK3!(在在f3函数中捕获异常函数中捕获异常)end3 (执行执行f3函数中最后一个语句时的输出函数中最后一个语句时的输出)end2 (执行执行f2函数中最后一个语句时的输出函数中最后一个语句时的输出)end1 (执行执行f1函数中最后一个语句时的输出函数中最后一个语句时的输出)end0 (执行主函数中最后一个语句时的输出执行主函

26、数中最后一个语句时的输出)(3) 如果在此基础上再将如果在此基础上再将f3函数中的函数中的catch块改为块改为catch(double) coutOK3!endl;throw;程序运行结果如下程序运行结果如下: OK3!(在在f3函数中捕获异常函数中捕获异常)OK0! (在主函数中捕获异常在主函数中捕获异常)end0 (执行主函数中最后一个语句时的输出执行主函数中最后一个语句时的输出)为便于阅读程序,使用户在看程序时能够知道所用为便于阅读程序,使用户在看程序时能够知道所用的函数是否会抛出异常信息以及异常信息可能的类的函数是否会抛出异常信息以及异常信息可能的类型,型,C+允许在声明函数时列出可

27、能抛出的异常类型,允许在声明函数时列出可能抛出的异常类型,如可以将例如可以将例14.1中第二个程序的第中第二个程序的第3行改写为行改写为double triangle(double,double,double) throw(double);表示表示triangle函数只能抛出函数只能抛出double类型的异常信息。类型的异常信息。如果写成如果写成double triangle(double,double,double) throw(int,double,float,char);则表示则表示triangle函数可以抛出函数可以抛出int,double,float或或char类类型的异常信息。异常

28、指定是函数声明的一部分,必型的异常信息。异常指定是函数声明的一部分,必须同时出现在函数声明和函数定义的首行中,否则须同时出现在函数声明和函数定义的首行中,否则在进行函数的另一次声明时,编译系统会报告在进行函数的另一次声明时,编译系统会报告“类类型不匹配型不匹配”。14.1.3 在函数声明中进行异常情况指定在函数声明中进行异常情况指定如果在声明函数时未列出可能抛出的异常类型,则如果在声明函数时未列出可能抛出的异常类型,则该函数可以抛出任何类型的异常信息。如例该函数可以抛出任何类型的异常信息。如例14.1中第中第2个程序中所表示的那样。个程序中所表示的那样。如果想声明一个不能抛出异常的函数,可以写

29、成以如果想声明一个不能抛出异常的函数,可以写成以下形式下形式:double triangle(double,double,double) throw();/throw无参数无参数这时即使在函数执行过程中出现了这时即使在函数执行过程中出现了throw语句,实际语句,实际上也并不执行上也并不执行throw语句,并不抛出任何异常信息,语句,并不抛出任何异常信息,程序将非正常终止。程序将非正常终止。如果在如果在try块块(或或try块中调用的函数块中调用的函数)中定义了类对象,中定义了类对象,在建立该对象时要调用构造函数。在执行在建立该对象时要调用构造函数。在执行try块块 (包括包括在在try块中调

30、用其他函数块中调用其他函数) 的过程中如果发生了异常,的过程中如果发生了异常,此时流程立即离开此时流程立即离开try块。这样流程就有可能离开该块。这样流程就有可能离开该对象的作用域而转到其他函数,因而应当事先做好对象的作用域而转到其他函数,因而应当事先做好结束对象前的清理工作,结束对象前的清理工作,C+的异常处理机制会在的异常处理机制会在throw抛出异常信息被抛出异常信息被catch捕获时,对有关的局部对捕获时,对有关的局部对象进行析构象进行析构(调用类对象的析构函数调用类对象的析构函数), 析构对象的顺析构对象的顺序与构造的顺序相反,然后执行与异常信息匹配的序与构造的顺序相反,然后执行与异

31、常信息匹配的catch块中的语句。块中的语句。14.1.4 在异常处理中处理析构函数在异常处理中处理析构函数例例14.3 在异常处理中处理析构函数。在异常处理中处理析构函数。这是一个为说明在异常处理中调用析构函数的示例,这是一个为说明在异常处理中调用析构函数的示例,为了清晰地表示流程,程序中加入了一些为了清晰地表示流程,程序中加入了一些cout语句,语句,输出有关的信息,以便对照结果分析程序。输出有关的信息,以便对照结果分析程序。#include #include using namespace std;class Studentpublic: Student(int n,string nam

32、)/定义构造函数定义构造函数coutconstructor-nendl;num=n;name=nam;Student( )coutdestructor-numendl;/定义析构函数定义析构函数 void get_data( ); /成员函数声明成员函数声明private:int num;string name; ;void Student:get_data( ) /定义成员函数定义成员函数if(num=0) throw num; /如如num=0,抛出抛出int型变量型变量num else coutnum nameendl; /若若num0,输出输出num,name coutin get_d

33、ata()endl; /输出信息,表示目前在输出信息,表示目前在get_data函数中函数中 void fun( )Student stud1(1101,Tan); /建立对象建立对象stud1stud1.get_data( ); /调用调用stud1的的get_data函数函数Student stud2(0,Li); /建立对象建立对象stud2stud2.get_data( ); /调用调用stud2的的get_data函数函数int main( )coutmain beginendl; /表示主函数开始了表示主函数开始了coutcall fun( )endl; /表示调用表示调用fun函

34、数函数try fun( ); /调用调用fun函数函数catch(int n) coutnum=n,error!endl; /表示表示num=0出错出错coutmain endendl; /表示主函数结束表示主函数结束return 0;程序运行结果如下程序运行结果如下:main begincall fun( )constructor-11011101 tanin get_data()constructor-0destructor-0destructor-1101num=0,error!main end在学习本书前面各章时,已经多次看到在程序中用在学习本书前面各章时,已经多次看到在程序中用了以下

35、语句了以下语句: using namespace std;这就是使用了命名空间这就是使用了命名空间std。在本节中将对它作较详在本节中将对它作较详细的介绍。细的介绍。14.2 命名空间命名空间命名空间是命名空间是ANSI C+引入的可以由用户命名的作用引入的可以由用户命名的作用域,用来处理程序中常见的同名冲突。域,用来处理程序中常见的同名冲突。在在C语言中定义了语言中定义了3个层次的作用域,即文件个层次的作用域,即文件(编译单编译单元元) 、函数和复合语句。、函数和复合语句。C+又引入了类作用域,类又引入了类作用域,类是出现在文件内的。在不同的作用域中可以定义相是出现在文件内的。在不同的作用域

36、中可以定义相同名字的变量,互不干扰,系统能够区别它们。同名字的变量,互不干扰,系统能够区别它们。下面先简单分析一下作用域的作用,然后讨论命名下面先简单分析一下作用域的作用,然后讨论命名空间的作用。空间的作用。如果在文件中定义了两个类,在这两个类中可以有如果在文件中定义了两个类,在这两个类中可以有同名的函数。在引用时,为了区别,应该加上类名同名的函数。在引用时,为了区别,应该加上类名作为限定,如作为限定,如14.2.1 为什么需要命名空间为什么需要命名空间class A/声明声明A类类 public:void fun1( ); /声明声明A类中的类中的fun1函数函数 private:int i

37、; ;void A:fun1( ) /定义定义A类中的类中的fun1函数函数 / class B /声明声明B类类 public:void fun1( ); /B类中也有类中也有fun1函数函数void fun2( ); ;void B:fun1( ) /定义定义B类中的类中的fun1函数函数 / 这样不会发生混淆。这样不会发生混淆。在文件中可以定义全局变量在文件中可以定义全局变量(global variable),它的它的作用域是整个程序。如果在文件作用域是整个程序。如果在文件A中定义了一个变量中定义了一个变量aint a=3;在文件在文件B中可以再定义一个变量中可以再定义一个变量aint

38、a=5;在分别对文件在分别对文件A和文件和文件B进行编译时不会有问题。但进行编译时不会有问题。但是,如果一个程序包括文件是,如果一个程序包括文件A和文件和文件B,那么在进行那么在进行连接时,会报告出错,因为在同一个程序中有两个连接时,会报告出错,因为在同一个程序中有两个同名的变量,认为是对变量的重复定义。问题在于同名的变量,认为是对变量的重复定义。问题在于全局变量的作用域是整个程序,在同一作用域中不全局变量的作用域是整个程序,在同一作用域中不应有两个或多个同名的实体应有两个或多个同名的实体(entity),包括变量、函包括变量、函数和类等。数和类等。 可以通过可以通过extern声明同一程序中

39、的两个文件中的同名声明同一程序中的两个文件中的同名变量是同一个变量。如果在文件变量是同一个变量。如果在文件B中有以下声明中有以下声明:extern int a;表示文件表示文件B中的变量中的变量a是在其他文件中已定义的变量。是在其他文件中已定义的变量。由于有此声明,在程序编译和连接后,文件由于有此声明,在程序编译和连接后,文件A的变量的变量a的作用域扩展到了文件的作用域扩展到了文件B。如果在文件如果在文件B中不再对中不再对a赋值,则在文件赋值,则在文件B中用以下语句输出的是文件中用以下语句输出的是文件A中变中变量量 a的值的值:couta;/得到得到a的值为的值为3在简单的程序设计中,只要人们

40、小心注意,可以争在简单的程序设计中,只要人们小心注意,可以争取不发生错误。但是,一个大型的应用软件,往往取不发生错误。但是,一个大型的应用软件,往往不是由一个人独立完成的,假如不同的人分别定义不是由一个人独立完成的,假如不同的人分别定义了类,放在不同的头文件中,在主文件了类,放在不同的头文件中,在主文件(包含主函数包含主函数的文件的文件)需要用这些类时,需要用这些类时,就用就用#include命令行将这些头文件包含进来。由于各命令行将这些头文件包含进来。由于各头文件是由不同的人设计的,有可能在不同的头文头文件是由不同的人设计的,有可能在不同的头文件中用了相同的名字来命名所定义的类或函数。这件中

41、用了相同的名字来命名所定义的类或函数。这样在程序中就会出现名字冲突。样在程序中就会出现名字冲突。例例14.4 名字冲突。名字冲突。程序员甲在头文件程序员甲在头文件header1.h中定义了类中定义了类Student和函和函数数fun。/header1.h (头文件头文件1,设其文件名为,设其文件名为cc14-4-h1.h)#include #include using namespace std;class Student/声明声明Student类类public: Student(int n,string nam,char s)num=n;name=nam;sex=s; void get_da

42、ta( );private:int num; string name; char sex; ;void Student:get_data( ) /成员函数定义成员函数定义coutnum name sexendl;double fun(double a,double b) /定义全局函数定义全局函数(即外部函数即外部函数)return sqrt(a+b);在在main函数所在的文件中包含头文件函数所在的文件中包含头文件header1.h:#include #include cc14-4-h1.h /注意要用双引号,因为文件一般是放在用户目录中的注意要用双引号,因为文件一般是放在用户目录中的usi

43、ng namespace std;int main( )Student stud1(101,Wang,18); /定义类对象定义类对象stud1 stud1.get_data( ); coutfun(5,3)endl; return 0; 程序能正常运行,输出为程序能正常运行,输出为101 Wang 182.82843如果程序员乙写了头文件如果程序员乙写了头文件header2.h,在其中除了定在其中除了定义其他类以外,还定义了类义其他类以外,还定义了类Student和函数和函数fun,但其但其内容与头文件内容与头文件header1.h中的中的Student和函数和函数fun有所有所不同。不同。

44、/header2.h (头文件头文件2,设其文件名为,设其文件名为cc14-4-h2.h)#include #include using namespace std;class Student/声明声明Student类类public: Student(int n,string nam,char s) /参数与参数与header1中的中的student不同不同num=n;name=nam;sex=s; void get_data( );private: int num; string name; char sex; /此项与此项与header1不同不同;void Student:get_data

45、( ) /成员函数定义成员函数定义coutnum name sexendl;double fun(double a,double b) /定义全局函数定义全局函数 return sqrt(a-b); /返回值与返回值与header1中的中的fun函数不同函数不同/头文件头文件2中可能还有其他内容中可能还有其他内容假如主程序员在其程序中要用到假如主程序员在其程序中要用到header1.h中的中的Student和函数和函数fun,因而在程序中包含了头文件因而在程序中包含了头文件header1.h,同时要用到头文件同时要用到头文件header2.h中的一些内中的一些内容,因而在程序中又包含了头文件容

46、,因而在程序中又包含了头文件header2.h。如果如果主文件主文件(包含主函数的文件包含主函数的文件)如下如下:/main file#include #include cc14-4-h1.h/包含头文件包含头文件1#include cc14-4-h2.h /包含头文件包含头文件2using namespace std;int main( )Student stud1(101,Wang,18);stud1.get_data();coutfun(5,3)endl;return 0; 这时程序编译就会出错。这时程序编译就会出错。 因为在预编译后,头文件因为在预编译后,头文件中的内容取代了对应的中的

47、内容取代了对应的#include命令行,这样就在同命令行,这样就在同一个程序文件中出现了两个一个程序文件中出现了两个Student类和两个类和两个fun函数,函数,显然是重复定义,这就是名字冲突,即在同一个作显然是重复定义,这就是名字冲突,即在同一个作用域中有两个或多个同名的实体。用域中有两个或多个同名的实体。不仅如此,在程序中还往往需要引用一些库,为此不仅如此,在程序中还往往需要引用一些库,为此需要包含有关的头文件。如果在这些库中包含有与需要包含有关的头文件。如果在这些库中包含有与程序的全局实体同名的实体,或者不同的库中有相程序的全局实体同名的实体,或者不同的库中有相同的实体名,则在编译时就

48、会出现名字冲突。同的实体名,则在编译时就会出现名字冲突。为了避免这类问题的出现,人们提出了许多方法,为了避免这类问题的出现,人们提出了许多方法,例如例如: 将实体的名字写得长一些;把名字起得特殊一将实体的名字写得长一些;把名字起得特殊一些,包括一些特殊的字符;由编译系统提供的内部些,包括一些特殊的字符;由编译系统提供的内部全局标识符都用下划线作为前缀,如全局标识符都用下划线作为前缀,如_complex(),以以避免与用户命名的实体同名;由软件开发商提供的避免与用户命名的实体同名;由软件开发商提供的实体的名字用特定的字符作为前缀。但是这样的效实体的名字用特定的字符作为前缀。但是这样的效果并不理想

49、,而且增加了阅读程序的难度,可读性果并不理想,而且增加了阅读程序的难度,可读性降低了。降低了。C语言和早期的语言和早期的C+语言没有提供有效的机制来解决语言没有提供有效的机制来解决这个问题,没有使库的提供者能够建立自己的命名这个问题,没有使库的提供者能够建立自己的命名空间的工具。人们希望空间的工具。人们希望ANSI C+标准能够解决这个标准能够解决这个问题,提供一种机制、一种工具,使由库的设计者问题,提供一种机制、一种工具,使由库的设计者命名的全局标识符能够和程序的全局实体名以及其命名的全局标识符能够和程序的全局实体名以及其他库的全局标识符区别开来。他库的全局标识符区别开来。为了解决上面这个问

50、题,为了解决上面这个问题,ANSI C+增加了命名空间增加了命名空间(namespace)。所谓命名空间,实际上就是一个由程所谓命名空间,实际上就是一个由程序设计者命名的内存区域。程序设计者可以根据需序设计者命名的内存区域。程序设计者可以根据需要指定一些有名字的空间域,把一些全局实体分别要指定一些有名字的空间域,把一些全局实体分别放在各个命名空间中,从而与其他全局实体分隔开放在各个命名空间中,从而与其他全局实体分隔开来。如来。如namespace ns1/指定命名空间指定命名空间ns1int a;double b;现在命名空间成员包括变量现在命名空间成员包括变量a和和b,注意注意a和和b仍然是

51、仍然是全局变量,仅仅是把它们隐藏在指定的命名空间中全局变量,仅仅是把它们隐藏在指定的命名空间中而已。而已。14.2.2 什么是命名空间什么是命名空间如果在程序中要使用变量如果在程序中要使用变量a和和b,必须加上命名空间必须加上命名空间名和作用域分辨符名和作用域分辨符“:”,如,如ns1:a,ns1:b。这种用这种用法称为命名空间限定法称为命名空间限定(qualified),这些名字这些名字(如如ns1:a)称为被限定名称为被限定名(qualified name)。C+中命名空间的作中命名空间的作用类似于操作系统中的目录和文件的关系。用类似于操作系统中的目录和文件的关系。命名空间的作用是建立一些

52、互相分隔的作用域,把命名空间的作用是建立一些互相分隔的作用域,把一些全局实体分隔开来,以免产生名字冲突。一些全局实体分隔开来,以免产生名字冲突。可以根据需要设置许多个命名空间,每个命名空间可以根据需要设置许多个命名空间,每个命名空间名代表一个不同的命名空间域,不同的命名空间不名代表一个不同的命名空间域,不同的命名空间不能同名。这样,可以把不同的库中的实体放到不同能同名。这样,可以把不同的库中的实体放到不同的命名空间中。过去我们用的全局变量可以理解为的命名空间中。过去我们用的全局变量可以理解为全局命名空间,独立于所有有名的命名空间之外,全局命名空间,独立于所有有名的命名空间之外,它是不需要用它是

53、不需要用namespace声明的,实际上是由系统隐声明的,实际上是由系统隐式声明的,存在于每个程序之中。式声明的,存在于每个程序之中。在声明一个命名空间时,花括号内不仅可以包括变在声明一个命名空间时,花括号内不仅可以包括变量,而且还可以包括以下类型量,而且还可以包括以下类型: 变量变量(可以带有初始化可以带有初始化);常量;常量;函数函数(可以是定义或声明可以是定义或声明);结构体;结构体;类;类;模板;模板;命名空间命名空间(在一个命名空间中又定义一个命名空间,在一个命名空间中又定义一个命名空间,即嵌套的命名空间即嵌套的命名空间)。例如例如namespace ns1const int RAT

54、E=0.08;/常量常量double pay; /变量变量double tax( ) /函数函数return a*RATE;namespace ns2 /嵌套的命名空间嵌套的命名空间 int age;如果想输出命名空间如果想输出命名空间ns1中成员的数据,可以采用下中成员的数据,可以采用下面的方法面的方法:coutns1:RATEendl;coutns1:payendl;coutns1:tax()endl;coutns1:ns2:ageendl;/需要指定外层的和内层的命名空间名需要指定外层的和内层的命名空间名现在,对例现在,对例14.4程序进行修改,使之能正确运行。程序进行修改,使之能正确运

55、行。例例14.5 利用命名空间来解决例利用命名空间来解决例14.4程序名字冲突问题。程序名字冲突问题。修改两个头文件,把在头文件中声明的类分别放在修改两个头文件,把在头文件中声明的类分别放在两个不同的命名空间中。两个不同的命名空间中。/header1.h (头文件头文件1)#include #include using namespace std;namespace ns1/声明命名空间声明命名空间ns1class Student /在命名空间在命名空间ns1内声明内声明Student类类public:Student(int n,string nam,int a)14.2.3 使用命名空间解决

56、名字冲突使用命名空间解决名字冲突num=n;name=nam;age=a;void get_data( ); private:int num;string name;int age; ; void Student:get_data() /定义成员函数定义成员函数 coutnum name ageendl; double fun(double a,double b) /在命名空间在命名空间ns1内定义内定义fun函数函数 return sqrt(a+b);/header2.h (头文件头文件2)#include #include using namespace std;namespace ns2

57、 /声明命名空间声明命名空间ns2 class Studentpublic:Student(int n,string nam,char s)num=n;name=nam;sex=s;void get_data( );private:int num;char name20;char sex;void Student:get_data( ) coutnum name sexendl;double fun(double a,double b)return sqrt(a-b); /main file (主文件主文件)#include #include cc14-5-h1.h /包含头文件包含头文件1#

58、include cc14-5-h2.h /包含头文件包含头文件2using namespace std;int main( )ns1:Student stud1(101,Wang,18); /用命名空间用命名空间ns1中声明的中声明的Student类定义类定义stud1stud1.get_data( ); /不要写成不要写成ns1:stud1.get_data( );coutns1:fun(5,3)endl; /调用命名空间调用命名空间ns1中的中的fun函数函数ns2:Student stud2(102,Li,f); /用命名空间用命名空间ns2中声明的中声明的Student类定义类定义st

59、ud2 stud2.get_data( );coutns2:fun(5,3)endl; /调用命名空间调用命名空间ns1中的中的fun函数函数return 0; 程序能顺利通过编译,并得到以下运行结果程序能顺利通过编译,并得到以下运行结果:101 Wang 18(对象对象stud1中的数据中的数据)2.82843 (5+3的开方值的开方值)102 Li f (对象对象stud2中的数据中的数据)1.41421 (5-3的开方值的开方值)在引用命名空间成员时,要用命名空间名和作用域在引用命名空间成员时,要用命名空间名和作用域分辨符对命名空间成员进行限定,以区别不同的命分辨符对命名空间成员进行限定

60、,以区别不同的命名空间中的同名标识符。即名空间中的同名标识符。即命名空间名命名空间名:命名空间成员名命名空间成员名这种方法是有效的,能保证所引用的实体有惟一的这种方法是有效的,能保证所引用的实体有惟一的名字。但是如果命名空间名字比较长,尤其在有命名字。但是如果命名空间名字比较长,尤其在有命名空间嵌套的情况下,为引用一个实体,需要写很名空间嵌套的情况下,为引用一个实体,需要写很长的名字。在一个程序中可能要多次引用命名空间长的名字。在一个程序中可能要多次引用命名空间成员,就会感到很不方便。成员,就会感到很不方便。为此,为此,C+提供了一些机制,能简化使用命名空间成提供了一些机制,能简化使用命名空间

61、成员的手续。员的手续。14.2.4 使用命名空间成员的方法使用命名空间成员的方法(1) 使用命名空间别名使用命名空间别名可以为命名空间起一个别名可以为命名空间起一个别名(namespace alias),用来用来代替较长的命名空间名。如代替较长的命名空间名。如namespace Television/声明命名空间,名为声明命名空间,名为Television 可以用一个较短而易记的别名代替它。如可以用一个较短而易记的别名代替它。如namespace TV = Television;/别名别名TV与原名与原名Television等价等价(2) 使用使用using 命名空间成员名命名空间成员名usi

62、ng后面的命名空间成员名必须是由命名空间限定后面的命名空间成员名必须是由命名空间限定的名字。例如的名字。例如using ns1:Student;using声明的有效范围是从声明的有效范围是从using语句开始到语句开始到using所在所在的作用域结束。如果在以上的的作用域结束。如果在以上的using语句之后有以下语句之后有以下语句语句: Student stud1(101,Wang,18);/此处的此处的Student相当于相当于ns1:Student上面的语句相当于上面的语句相当于ns1:Student stud1(101,Wang,18);又如又如using ns1:fun;/声明其后出现

63、的声明其后出现的fun是属于命名空间是属于命名空间ns1中的中的funcoutfun(5,3)endl; /此处的此处的fun函数相当于函数相当于ns1:fun(5,3)显然,这可以避免在每一次引用命名空间成员时都显然,这可以避免在每一次引用命名空间成员时都用命名空间限定,使得引用命名空间成员方便易用。用命名空间限定,使得引用命名空间成员方便易用。但是要注意但是要注意: 在同一作用域中用在同一作用域中用using声明的不同命名声明的不同命名空间的成员中不能有同名的成员。例如空间的成员中不能有同名的成员。例如using ns1:Student;/声明其后出现的声明其后出现的Student是命名空

64、间是命名空间ns1中的中的Studentusing ns2:Student; /声明其后出现的声明其后出现的Student是命名空间是命名空间ns2中的中的StudentStudent stud1; /请问此处的请问此处的Student是哪个命名空间中的是哪个命名空间中的Student?产生了二义性,编译出错。产生了二义性,编译出错。(3) 使用使用using namespace 命名空间名命名空间名能否在程序中用一个语句就能一次声明一个命名空能否在程序中用一个语句就能一次声明一个命名空间中的全部成员呢?间中的全部成员呢?C+提供了提供了using namespace语句来实现这一目的。语句来

65、实现这一目的。using namespace语句的一般格式为语句的一般格式为using namespace命名空间名;命名空间名;例如例如using namespace ns1;声明了在本作用域中要用到命名空间声明了在本作用域中要用到命名空间ns1中的成员,中的成员,在使用该命名空间的任何成员时都不必用命名空间在使用该命名空间的任何成员时都不必用命名空间限定。如果在作了上面的声明后有以下语句限定。如果在作了上面的声明后有以下语句:Student stud1(101,Wang,18);/Student隐含指命名空间隐含指命名空间ns1中的中的Studentcoutfun(5,3)endl; /这

66、里的这里的fun函数是命名空间函数是命名空间ns1中的中的fun函数函数在用在用using namespace声明的作用域中,命名空间声明的作用域中,命名空间ns1的成员就好像在全局域声明的一样。因此可以不必的成员就好像在全局域声明的一样。因此可以不必用命名空间限定。显然这样的处理对写程序比较方用命名空间限定。显然这样的处理对写程序比较方便。但是如果同时用便。但是如果同时用using namespace声明多个命名声明多个命名空间时,往往容易出错。因此只有在使用命名空间空间时,往往容易出错。因此只有在使用命名空间数量很少,以及确保这些命名空间中没有同名成员数量很少,以及确保这些命名空间中没有同名成员时才用时才用using namespace语句。语句。以上介绍的是有名字的命名空间,以上介绍的是有名字的命名空间,C+还允许使用没还允许使用没有名字的命名空间,如在文件有名字的命名空间,如在文件A中声明了以下的无名中声明了以下的无名命名空间命名空间:namespace/命名空间没有名字命名空间没有名字void fun() /定义命名空间成员定义命名空间成员 coutOK.endl;由于命名空

展开阅读全文
温馨提示:
1: 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
2: 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
3.本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
5. 装配图网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。
关于我们 - 网站声明 - 网站地图 - 资源地图 - 友情链接 - 网站客服 - 联系我们

copyright@ 2023-2025  zhuangpeitu.com 装配图网版权所有   联系电话:18123376007

备案号:ICP2024067431-1 川公网安备51140202000466号


本站为文档C2C交易模式,即用户上传的文档直接被用户下载,本站只是中间服务平台,本站所有文档下载所得的收益归上传人(含作者)所有。装配图网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对上载内容本身不做任何修改或编辑。若文档所含内容侵犯了您的版权或隐私,请立即通知装配图网,我们立即给予删除!