27

我被问了一个问题,并被要求给出输出。

int main(void){  
    int x = 2;  
    switch(x){  
        case 1,2,1: printf("Case 1 is executed");  
            break;  
        case 2,3,1: printf("Case 2 is executed");  
            break;  
        default : printf("Default case us executed");  
    }  
    return 0;  
}

上面的代码在 Turbo C 中给出了“案例 1 已执行”的输出,但是在代码块和在线编译上,它给出了编译器错误。

哪一个是正确的?是不是编译器错误?如果不是,为什么代码只能在 Turbo C 上运行?

4

6 回答 6

38

是否是编译器错误。

该代码在两种语言中都无效:case表达式必须是常量表达式,并且常量表达式不能包含逗号运算符。(在 C 中,这是明确说明的;在 C++ 中,您必须取消选择语法才能发现常量表达式必须是条件表达式,不能包含逗号)。

即使您被允许在此处使用逗号运算符,该switch语句仍然无效,因为两种情况都具有相同的值 1。

如果不是,为什么代码只能在 Turbo C 上运行。

因为自从上次更新史前编译器以来,两种语言都发生了重大变化。如果您想学习本世纪以来的 C 或 C++ 变体,请不要使用它。

于 2013-10-19T14:16:09.867 回答
10

switch 语句中的逗号运算符是什么意思?
这意味着你有一个旧的编译器。

编辑帖子(显示case range示例)

前两个示例(包括您的原始代码)显示了不正确的 switch 语句语法(带有解释)。第三个代码示例显示了如何正确堆叠大小写标签:

在您的代码中,编译器应该在 case 1,<--之后标记了第一个逗号

#include <ansi_c.h>
int main(void){  
    int x = 2;  
    switch(x)
    {  
        case 1,2,1: printf("Case 1 is executed");  
        break;  //error flagged at first comma, and all comma after in case
        case 2,3,1: printf("Case 2 is executed");  
        break;  
        default : printf("Default case is executed");  
    }  
    return 0;  
}  

而且,即使这样修改,您也应该得到重复标签错误:

#include <ansi_c.h>
int main(void){  
    int x = 2;  
    switch(x)
    {  
        case 1:
        case 2:
        case 1: printf("Case 1 is executed"); //duplicate label 1 error. (and others below) 
            break;  
        case 2:
        case 3:
        case 1: printf("Case 2 is executed");  
            break;

        default : printf("Default case is executed");  
    }
    return 0;  
}

这个例子是完全合法的(C99,C11)并且有用:即,没有重复的标签,并且语法通过堆叠唯一标签来处理case 1: OR case 2: OR case 3:应该以相同方式处理的条件(在同一个块中),从而符合正确的开关用法. 当然,情况 4、5 和 6 也是如此。

#include <ansi_c.h>
int main(void){  
    int x = 2;  
    switch(x)
    {  
        case 1:
        case 2:
        case 3: printf("Case 1,2 or 3 is executed"); //duplicate label 1 error. (and others below) 
            break;  
        case 4:
        case 5:
        case 6: printf("Case 4,5 or 6 is executed");  
            break;
    }
    getchar();
    return 0;  
}

最后一个示例只是为了完整性。它说明了case range表达式。尽管引起了 C 程序员的兴趣,但它还不是 C99 或 C11 的一部分,而是 Sun(一种 unix 风格)GNU C 编译器 (等)的扩展:

...
    switch(x)
    {  
            case 'a' ... 'z':  //note: spaces between all characters ('a') and ellipses are required
                    printf("lowercase alpha char detected");
                    break;
            case 'A' ... 'B':
                    printf("uppercase alpha char detected");
                    break;

            default: printf("Default case is executed");  
    }
...

您看到从一个编译器到另一个编译器的模棱两可结果的原因可能是 Turbo C 真的老了。您正在使用的版本可能是针对不再适用的 C 标准版本实施的。

考虑更改为当前的编译器。一个便宜(免费)的替代品是MinGW。MinGW 是一个维护良好的开源编译器。如果您喜欢使用集成开发环境 (IDE),Code::Blocks是一个选项,也是免费的,并且作为选项与 MinGW 捆绑在一起。

关于兼容性,请在此链接中搜索与其他编译器套件的比较以了解 MinGW 扩展。MinGW 扩展在扩展功能的同时,有时会使使用它们编写的代码与其他当前编译器不可移植。建议在使用时谨慎使用。

于 2013-10-19T14:21:12.627 回答
1

Turbo C 在 switch case 上使用逗号运算符并取最后一个值,例如 case 1, 2, 3: 将编译为 case 3: case 2, 3, 1 as case 1: 因此 Turbo C 不会给您任何错误。其他编译器不允许您使用 case 1, 2, 3: kind 语句本身。

但在你的情况下,即使 Turbo c 也会出错,因为 case 语句类似于 case 1, 2, 1: 和 case 3, 2, 1: 将被编译为 case 1: 和 case 1: 因此根据 switch case 规则您只能有 1 个带有值的案例,并且您不能重复该案例

我更喜欢使用 gcc 编译器而不是 Turbo C

于 2013-10-19T14:48:17.067 回答
0

仅供参考,今天人们可以在适当的时候(不经常)使用GCC 案例范围扩展来获得类似的结果(仅支持范围,而不是任意的值列表)。

case 1 ... 5:
case 'A' ... 'Z':
于 2013-10-25T11:50:29.750 回答
0

是否是编译器错误。

它会导致编译错误(不知道TURBO C++,但在现代编译器中)。
这不是 switch 语句的工作方式(c/c++ 中的语法无效)。您不能在语句中重用case值(请参阅chris给出的链接)您可以这样做;switch

 switch(x){
            case 1: case 2: case 3: printf("Case 1 is executed");
            break;
            default : printf("Default case us execyted"); 
          }
于 2013-10-19T14:06:15.047 回答
0

您不能使用两次相同的“案例”值
这是不正确的:case 1: case 2: case 1: printf("Case 1 is executed");
尝试在 VS2010 上编译 =>(错误 C2196:案例值“1”已使用)

于 2013-10-19T14:12:55.257 回答