韩冰 发表于 2005-1-25 17:57

概观C++程序设计语言(大规模程序设计)

<TABLE width="100%" border=0>

<TR>
<TD width="7%"> </TD>
<TD width="85%">
<DIV class=Section1 style="LAYOUT-GRID:  15.6pt none">
<P  align=center></P>
<P ><B>5          </B><B>大规模程序设计</B>(Large-scale Programming)<B></B></P>
<P><B> </B></P>
<P >本文的主要内容涵盖的是C++所支持的主要程序设计风格。然而namespace和异常处理机制虽然是重要的语言特性,但却并不属于这种讨论范畴,因为它们在所有的程序设计风格中都能支持大规模程序设计。在将分散的部件组合成完整程序的过程中,namespace和异常处理机制被用来缓减整个过程的难度和复杂度;随着程序规模的增大,它们也随之起到越来越重要的作用。</P>
<P> </P>
<P><B>5</B><B>.</B><B>1  </B><B>异常和错误处理</B>(Exceptions and Error Handling)</P>
<P >当在某个位置检测到一个错误的时候,我们使用<I>异常</I>来把程序的控制权转交到能够处理这种错误的调用者(caller)那里。显然,这种机制只用来处理那些在当前区域内无法处理的错误。</P>
<P >有人可能会问:“如果一个位置所发生的事情最终能够被正确的处理并使得程序如期正常运行,那么这件事情又怎么能被认为是一个错误呢?”事实上,我们把这类异常事件(或简称异常)以及用来处理这类事件的语言机制一起成为异常处理(exception handling)。</P>
<P>我们可以像这样报告Stack中发生的上溢(overflow)和下溢(underflow)错误:</P>
<P>template&lt;class T&gt; class Stack {</P>
<P >T* v;</P>
<P >Int max_size;</P>
<P >Int top;</P>
<P>Public:</P>
<P >class Underflow { };     // 用来报告下溢情况的型别</P>
<P >class Overflow { };      // 用来报告上溢情况的型别</P>
<P >Stack(int s); // 构造函数</P>
<P >~Stack();     // 析构函数</P>
<P >void push(T c)</P>
<P >{</P>
<P >    if (top == max_size) throw Overflow(); // 检查是否发生错误</P>
<P >    v = c; // 增加top并将c存放在top位置</P>
<P >}</P>
<P >T pop()</P>
<P >{</P>
<P >    if (top == 0) throw Underflow(); // 检查是否发生错误</P>
<P >    return v[--top]; //减少top</P>
<P >}</P>
<P>};</P>
<P>一个被抛出的对象能够被抛出该对象的那个函数中的相关调用者(caller)捕获。例如:</P>
<P>void f()</P>
<P>{</P>
<P >Stack&lt;string&gt; ss (10);</P>
<P >Try {</P>
<P >    ss.push(“Quiz”);</P>
<P >    string s = ss.pop(); // 用pop操作弹出”Quiz”</P>
<P >    ss.pop(); // 这一句企图对一个空的string施以pop操作,将会抛出Underflow</P>
<P >}</P>
<P >catch (Stack&lt;string&gt;::Underflow) { // 异常处理器</P>
<P >    cerr &lt;&lt; “error: stack underflow”;</P>
<P >    return;</P>
<P >}</P>
<P >catch (Stack&lt;string&gt;::Overflow) { // 异常处理器</P>
<P >    cerr &lt;&lt; “error: stack overflow”;</P>
<P >    return;</P>
<P >}</P>
<P >// …</P>
<P>}</P>
<P>在一个try{…}代码段中抛出的异常,或者在try{…}代码段所调用的一段代码中抛出的异常会被其对应型别的catch语句捕获。</P>
<P >我们可以使用异常来把错误处理变得更具独特性且更自然。特别是异常类的层次结构,它可以用来将异常进行分组,从而使一段处理代码能够只在适当的细节层级上处理错误。例如,我们假设open_file_error类派生自io_error:</P>
<P>void use_file(const char* fn)</P>
<P>{</P>
<P >File_ptr f(fn, “r”); // 打开fn以便读取;</P>
<P >// 如果在这个过程中出现错误,则抛出open_file_error</P>
<P >// 这里的代码使用fn</P>
<P>}</P>
<P>void some_fct()</P>
<P>{</P>
<P >try {</P>
<P >    // …</P>
<P >    use_file(“homedir/magic”);</P>
<P >    // …</P>
<P >}</P>
<P >catch (io_error) {</P>
<P >    // …</P>
<P >}</P>
<P>}</P>
<P>代码中的some_fct并不需要了解具体是什么出错了;这即是说,它并不需要认识open_file_error。some_fct只处理io_error这个抽象层级上的错误。</P>
<P >可以注意到,use_file本身根本不处理异常。其任务只是打开一个文件并予以使用。一旦没有可以打开的文件,程序控制权立即被转交给了some_fct。同样,一旦在使用文件的过程中发生了读取方面的错误,该文件将会在控制权转到some_fct之前经由File_ptr的析构函数而被正常关闭。</P>
<P> </P>
<P><B>5</B><B>.</B><B>2  namespaces</B><B>(名字空间)</B><B></B></P>
<P >一个namespace是指一个具名的范围(named scope)。namespace被用来将相关的声明划归在一起,将不相关的代码部分隔开。例如,两个各自独立开发的程序库使用相同的名称来代表不同的东西,但用户还是能够同时使用它们:</P>
<P>namespace Mylib {</P>
<P >template&lt;class T&gt; class Stack { /* … */ };</P>
<P >// …</P>
<P>}</P>
<P>namespace Yourlib {</P>
<P >class Stack { /* … */ };</P>
<P >// …</P>
<P>}</P>
<P>void f(int max)</P>
<P>{</P>
<P >Mylib::Stack&lt;int&gt; s1(max); // 使用我的那个Stack</P>
<P >Yourlib::Stack s2(max);     // 使用你的那个Stack</P>
<P >// …</P>
<P>}</P>
<P>然而,让一个namespace名称在同一处不断的重复出现,反而会影响和迷惑阅读者和编写者自己。因此,C++规定:在一个特定的namespace内部,即使不做显式的限定,也默认为该namespace中所包含的对象名称是有效的。例如:</P>
<P>void f(int max)</P>
<P>{</P>
<P >using namespace Mylib;   // 使Mylib()中的各种名称成为有效可用的</P>
<P >Stack&lt;int&gt; s1(max);      // 使用我的Stack</P>
<P >Yourlib::Stack s2(max); // 使用你的Stack</P>
<P >// …</P>
<P>}</P>
<P>namespace为管理不同的程序库和不同版本的代码提供了一个强有力的工具;特别是,它能让程序员自己控制一个non-local名称之引用的能见度。</P>
<P></P></DIV></TD>
<TD width="8%"> </TD></TR></TABLE>
页: [1]
查看完整版本: 概观C++程序设计语言(大规模程序设计)