为什么在函数末尾放置标签时必须包含分号?

Gov*_*mar 3 c label goto

带有goto错误处理程序的简单函数:

void checkFile(char *name)
{
    FILE *fp = fopen(name, "r");
    if(NULL == fp)
    {
        fprintf(stderr, "Could not open %s\n", name);
        goto end;
    }

    // use fp...

    fclose(fp);
    end:
    ;
}
Run Code Online (Sandbox Code Playgroud)

请注意,如果我end:在函数编译失败后删除了无用的分号。

在海湾合作委员会上:

error: expected primary-expression before '}' token
   16 | }
Run Code Online (Sandbox Code Playgroud)

在 MSVC 上:

error C2143: syntax error: missing ';' before '}'
Run Code Online (Sandbox Code Playgroud)

所以,我知道 C 标准确实说goto关键字的目标需要第 6.8.6.1 节 p 2 中的语句:

goto 语句会导致无条件跳转到封闭函数中以命名标签为前缀语句

但是,错误存在只是因为标签存在;如果我删除goto关键字,标签本身仍然被视为错误并且不会编译。我阅读了关于“带标签的语句”(第 6.8.1 节)的标准部分,但仍然没有找到任何可以解释这种奇怪约束的内容。

Vla*_*cow 5

在 C 中,标签可以放在语句之前。因此,如果没有语句,您可以放置​​一个空语句。

来自 C 标准(6.8.1 标记语句)

labeled-statement:
    identifier : statement
    case constant-expression : statement
    default : statement
Run Code Online (Sandbox Code Playgroud)

And(6.8.3 表达式和空语句)

expression-statement:
    expressionopt ;
Run Code Online (Sandbox Code Playgroud)

3 空语句(仅由分号组成)不执行任何操作。

在 C++ 声明中与 C 相对的是语句。因此,在 C++ 中,您可以在声明之前放置一个标签。

这里是演示程序。

C程序。

#include <stdio.h>

int main(void) 
{
    goto End;
    
    End:;
    const char *bye = "Bye";
    
    puts( bye );
    
    return 0;
}
Run Code Online (Sandbox Code Playgroud)

程序输出是

Bye
Run Code Online (Sandbox Code Playgroud)

C++程序

#include <iostream>

int main() 
{
    goto End;
    
    End:
    const char *bye = "Bye";
    
    std::cout <<  bye << '\n';

    return 0;
}
Run Code Online (Sandbox Code Playgroud)

程序输出是

Bye
Run Code Online (Sandbox Code Playgroud)

注意在C程序中label后面放了一个null语句

End:;
Run Code Online (Sandbox Code Playgroud)

没有它,编译器会报错。