Skip to content

第5章 语句

简单语句(Simple Statements)

如果在程序的某个地方,语法上需要一条语句但是逻辑上不需要,则应该使用空语句(null statement)。空语句中只含有一个单独的分号;

// read until we hit end-of-file or find an input equal to sought
while (cin >> s && s != sought)
    ;   // null statement

使用空语句时应该加上注释,从而令读这段代码的人知道该语句是有意省略的。

多余的空语句并非总是无害的。

// disaster: extra semicolon: loop body is this null statement
while (iter != svec.end()) ;    // the while body is the empty statement
    ++iter;     // increment is not part of the loop

复合语句(compound statement)是指用花括号括起来的(可能为空)语句和声明的序列。复合语句也叫做块(block),一个块就是一个作用域。在块中引入的名字只能在块内部以及嵌套在块中的子块里访问。通常,名字在有限的区域内可见,该区域从名字定义处开始,到名字所在(最内层)块的结尾处为止。

语句块不以分号作为结束。

空块的作用等价于空语句。

语句作用域(Statement Scope)

可以在ifswitchwhilefor语句的控制结构内定义变量,这些变量只在相应语句的内部可见,一旦语句结束,变量也就超出了其作用范围。

while (int i = get_num())   // i is created and initialized on each iteration
    cout << i << endl;
i = 0;  // error: i is not accessible outside the loop

条件语句(Conditional Statements)

if语句(The if Statement)

if语句的形式:

if (condition)
{
    statement;
}

if-else语句的形式:

if (condition)
{
    statement 1;
}
else
{
    statement 2;
}

其中condition是判断条件,可以是一个表达式或者初始化了的变量声明。condition必须用圆括号括起来。

  • 如果condition为真,则执行statement。执行完成后,程序继续执行if语句后面的其他语句。

  • 如果condition为假,则跳过statement。对于简单if语句来说,程序直接执行if语句后面的其他语句;对于if-else语句来说,程序先执行statement2,再执行if语句后面的其他语句。

if语句可以嵌套,其中else与离它最近的尚未匹配的if相匹配。

switch语句(The switch Statement)

switch语句的形式:

switch (number)
{
    case 1:
    {
        statement 1;
        break;
    }
    case 2:
    {
        statement 2;
        break;
    }
    default:
    {
        statement 3;
        break;
    }
}

switch语句先对括号里的表达式求值,值转换成整数类型后再与每个case标签(case label)的值进行比较。如果表达式的值和某个case标签匹配,程序从该标签之后的第一条语句开始执行,直到到达switch的结尾或者遇到break语句为止。case标签必须是整型常量表达式。

通常情况下每个case分支后都有break语句。如果确实不应该出现break语句,最好写一段注释说明程序的逻辑。

尽管switch语句没有强制要求在最后一个case标签后写上break,但为了安全起见,最好添加break。这样即使以后增加了新的case分支,也不用再在前面补充break语句了。

switch语句中可以添加一个default标签(default label),如果没有任何一个case标签能匹配上switch表达式的值,程序将执行default标签后的语句。

即使不准备在default标签下做任何操作,程序中也应该定义一个default标签。其目的在于告诉他人我们已经考虑到了默认情况,只是目前不需要实际操作。

不允许跨过变量的初始化语句直接跳转到该变量作用域内的另一个位置。如果需要为switch的某个case分支定义并初始化一个变量,则应该把变量定义在块内。

case true:
{
    // ok: declaration statement within a statement block
    string file_name = get_file_name();
    // ...
}

迭代语句(Iterative Statements)

迭代语句通常称为循环,它重复执行操作直到满足某个条件才停止。whilefor语句在执行循环体之前检查条件,do-while语句先执行循环体再检查条件。

while语句(The while Statement)

while语句的形式:

while (condition)
{
    statement;
}

只要condition的求值结果为true,就一直执行statement(通常是一个块)。condition不能为空,如果condition第一次求值就是falsestatement一次都不会执行。

定义在while条件部分或者循环体内的变量每次迭代都经历从创建到销毁的过程。

在不确定迭代次数,或者想在循环结束后访问循环控制变量时,使用while比较合适。

传统的for语句(Traditional for Statement)

for语句的形式:

for (initializer; condition; expression)
{
    statement;
}

一般情况下,initializer负责初始化一个值,这个值会随着循环的进行而改变。condition作为循环控制的条件,只要condition的求值结果为true,就执行一次statement。执行后再由expression负责修改initializer初始化的变量,这个变量就是condition检查的对象。如果condition第一次求值就是falsestatement一次都不会执行。initializer中也可以定义多个对象,但是只能有一条声明语句,因此所有变量的基础类型必须相同。

for语句头中定义的对象只在for循环体内可见。

范围for语句(Range for Statement)

范围for语句的形式:

for (declaration : expression)
{
    statement;
}

其中expression表示一个序列,拥有能返回迭代器的beginend成员。declaration定义一个变量,序列中的每个元素都应该能转换成该变量的类型(可以使用auto)。如果需要对序列中的元素执行写操作,循环变量必须声明成引用类型。每次迭代都会重新定义循环控制变量,并将其初始化为序列中的下一个值,之后才会执行statement

do-while语句(The do-while Statement)

do-while语句的形式:

do
{
    statement;
}
while (condition);

计算condition的值之前会先执行一次statementcondition不能为空。如果condition的值为false,循环终止,否则重复执行statement

因为do-while语句先执行语句或块,再判断条件,所以不允许在条件部分定义变量。

跳转语句(Jump Statements)

跳转语句中断当前的执行过程。

break语句(The break Statement)

break语句只能出现在迭代语句或者switch语句的内部,负责终止离它最近的whiledo-whilefor或者switch语句,并从这些语句之后的第一条语句开始执行。

string buf;
while (cin >> buf && !buf.empty())
{
    switch(buf[0])
    {
        case '-':
            // process up to the first blank
            for (auto it = buf.begin()+1; it != buf.end(); ++it)
            {
                if (*it == ' ')
                break;  // #1, leaves the for loop
                // . . .
            }
            // break #1 transfers control here
            // remaining '-' processing:
            break;  // #2, leaves the switch statement
        case '+':
    // . . .
    } // end switch
// end of switch: break #2 transfers control here
} // end while

continue语句(The continue Statement)

continue语句只能出现在迭代语句的内部,负责终止离它最近的循环的当前一次迭代并立即开始下一次迭代。和break语句不同的是,只有当switch语句嵌套在迭代语句内部时,才能在switch中使用continue

continue语句中断当前迭代后,具体操作视迭代语句类型而定:

  • 对于whiledo-while语句来说,继续判断条件的值。

  • 对于传统的for语句来说,继续执行for语句头中的第三部分,之后判断条件的值。

  • 对于范围for语句来说,是用序列中的下一个元素初始化循环变量。

goto语句(The goto Statement)

goto语句(labeled statement)是一种特殊的语句,在它之前有一个标识符和一个冒号。

end: return; // labeled statement; may be the target of a goto

标签标识符独立于变量和其他标识符的名字,它们之间不会相互干扰。

goto语句的形式:

goto label;

goto语句使程序无条件跳转到标签为label的语句处执行,但两者必须位于同一个函数内,同时goto语句也不能将程序的控制权从变量的作用域之外转移到作用域之内。

建议不要在程序中使用goto语句,它使得程序既难理解又难修改。

try语句块和异常处理(try Blocks and Exception Handling)

异常(exception)是指程序运行时的反常行为,这些行为超出了函数正常功能的范围。当程序的某一部分检测到一个它无法处理的问题时,需要使用异常处理(exception handling)。

异常处理机制包括throw表达式(throw expression)、try语句块(try block)和异常类(exception class)。

  • 异常检测部分使用throw表达式表示它遇到了无法处理的问题(throw引发了异常)。

  • 异常处理部分使用try语句块处理异常。try语句块以关键字try开始,并以一个或多个catch子句(catch clause)结束。try语句块中代码抛出的异常通常会被某个catch子句处理,catch子句也被称作异常处理代码(exception handler)。

  • 异常类用于在throw表达式和相关的catch子句之间传递异常的具体信息。

throw表达式(A throw Expression)

throw表达式包含关键字throw和紧随其后的一个表达式,其中表达式的类型就是抛出的异常类型。

try语句块(The try Block)

try语句块的通用形式:

try
{
    program-statements;
}
catch (exception-declaration)
{
    handler-statements;
}
catch (exception-declaration)
{
    handler-statements;
}

try语句块中的program-statements组成程序的正常逻辑,其内部声明的变量在块外无法访问,即使在catch子句中也不行。catch子句包含关键字catch、括号内一个对象的声明(异常声明,exception declaration)和一个块。当选中了某个catch子句处理异常后,执行与之对应的块。catch一旦完成,程序会跳过剩余的所有catch子句,继续执行后面的语句。

如果最终没能找到与异常相匹配的catch子句,程序会执行名为terminate的标准库函数。该函数的行为与系统有关,一般情况下,执行该函数将导致程序非正常退出。类似的,如果一段程序没有try语句块且发生了异常,系统也会调用terminate函数并终止当前程序的执行。

标准异常(Standard Exceptions)

异常类分别定义在4个头文件中:

  • 头文件exception定义了最通用的异常类exception。它只报告异常的发生,不提供任何额外信息。

  • 头文件stdexcept定义了几种常用的异常类。

类型 含义
exception 最常见的错误
runtime_error 运行时错误
overflow_error 计算上溢
underflow_error 计算下溢
range_error 数值超出有意义的值域
logic_error 程序逻辑错误
domain_error 参数值不存在
invalid_argument 无效参数
out_of_range 试图创建超出该类型最大长度的对象
length_error 数值超出有效范围
  • 头文件new定义了bad_alloc异常类。

  • 头文件type_info定义了bad_cast异常类。

标准库异常类的继承体系:

classDiagram
    exception <|-- runtime_error
    exception <|-- bad_cast
    exception <|-- bad_alloc
    exception <|-- logic_error
    runtime_error <|-- overflow_error
    runtime_error <|-- underflow_error
    runtime_error <|-- range_error
    logic_error <|-- domain_error
    logic_error <|-- invalid_argument
    logic_error <|-- out_of_range
    logic_error <|-- length_error

只能以默认初始化的方式初始化exceptionbad_allocbad_cast对象,不允许为这些对象提供初始值。其他异常类的对象在初始化时必须提供一个string或一个C风格字符串,通常表示异常信息。what成员函数可以返回该字符串的string副本。