"Key Takeaway"

考虑这样一种情况,我们希望向用户显示一个菜单,并要求他们进行选择——如果用户选择了无效的选择,则需要再次询问。显然,菜单和选项应该放在某种循环中(这样才可以反复询问用户,直到他们输入有效的输入),但是应该选择哪种类型的循环呢?

由于while循环会先对条件求值,所以使用while循环很别扭。假设我们这样做:

#include <iostream>
 
int main()
{
    // 选择必须被定义在循环外
    int selection{ 0 };
 
    while (selection != 1 && selection != 2 &&
        selection != 3 && selection != 4)
    {
        std::cout << "Please make a selection: \n";
        std::cout << "1) Addition\n";
        std::cout << "2) Subtraction\n";
        std::cout << "3) Multiplication\n";
        std::cout << "4) Division\n";
        std::cin >> selection;
    }
 
    // 继续选择进行操作
    // 可以使用 switch 语句进行
 
    std::cout << "You selected option #" << selection << '\n';
 
    return 0;
}

但是,上述代码能够正确工作的原因是因为此处的初始值为0——它不是任何合法的selection值 (1, 2, 3 or 4)。如果 0 也是合法的值会怎样?我们必须选择一个不同的初始化值来表示“无效”——这导致我们的代码中引入魔鬼数字(4.15 -字面量)。

我们可以添加一个新的变量来跟踪有效性,就像这样:

#include <iostream>
 
int main()
{
    int selection { 0 };
    bool invalid { true }; // new variable just to gate the loop
 
    while (invalid)
    {
        std::cout << "Please make a selection: \n";
        std::cout << "1) Addition\n";
        std::cout << "2) Subtraction\n";
        std::cout << "3) Multiplication\n";
        std::cout << "4) Division\n";
 
        std::cin >> selection;
        invalid = (selection != 1 && selection != 2 &&
            selection != 3 && selection != 4);
    }
 
    // do something with selection here
    // such as a switch statement
 
    std::cout << "You selected option #" << selection << '\n';
 
    return 0;
}

虽然这避免了魔鬼数字,但引入了一个新变量,以确保循环能够至少运行一次,这增加了复杂性和犯错的可能性。

do-while 语句

为了帮助解决上述问题,C++提供了do-while语句:

do
    statement; // 可以是单一语句,也可以是复合句
while (condition);

do-while语句是一个循环构造方式,其工作原理与while循环类似,只不过该语句总是至少执行一次。语句执行后,do-while循环检查条件。如果条件的计算结果为true ,执行路径将跳转回do-while循环的顶部并再次执行。

下面是我们上面用do-while循环代替while循环的例子:

#include <iostream>
 
int main()
{
    // selection must be declared outside of the do-while so we can use it later
    int selection{};
 
    do
    {
        std::cout << "Please make a selection: \n";
        std::cout << "1) Addition\n";
        std::cout << "2) Subtraction\n";
        std::cout << "3) Multiplication\n";
        std::cout << "4) Division\n";
        std::cin >> selection;
    }
    while (selection != 1 && selection != 2 &&
        selection != 3 && selection != 4);
 
    // do something with selection here
    // such as a switch statement
 
    std::cout << "You selected option #" << selection << '\n';
 
    return 0;
}

通过这种方式,我们既避免了魔鬼数字,也避免了额外的变量。

在上面的例子中有一件事值得讨论,那就是selection 变量必须在do块之外声明。如果在do块中声明了 selection 变量,那么当do块终止时,它将被销毁,这发生在条件值被求值之前。但是我们需要在while条件语句中声明变量——因此,selection 变量必须在do块之外声明(即使它没有在函数体后面使用)。

实际上,do-while循环并不常用。将条件放在循环的底部会模糊循环条件,这可能会导致错误。因此,许多开发人员建议完全避免do-while循环。我们将采取一种较为温和的立场,如果两种方法都可以使用时,使用while循环而不是do-while循环。

"最佳实践"

当给出相同的选择时,更倾向于while循环而不是do-while循环。