注释掉未使用的行后,开关案例将无法编译

Kor*_*gay 82 c

这是我的代码:

#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <netdb.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <arpa/inet.h>

int main (void) {

  struct addrinfo hints; 
  memset (&hints, 0, sizeof hints);

  hints.ai_family = AF_UNSPEC; 
  hints.ai_socktype = SOCK_DGRAM;  
  hints.ai_flags = AI_CANONNAME;   

  struct addrinfo *res;

  getaddrinfo ("example.com", "http", &hints, &res);
  printf ("Host: %s\n", "example.com");

  void *ptr;

  while (res != NULL) {
    printf("AI Family for current addrinfo: %i\n", res->ai_family);
    switch (res->ai_family) {
      case AF_INET:
        ptr = (struct sockaddr_in *) res->ai_addr;
        struct sockaddr_in *sockAddrIn = (struct sockaddr_in *) res->ai_addr;
        break;
    }
    res = res->ai_next;
  }
  return 0;
}
Run Code Online (Sandbox Code Playgroud)

编译好.

但是,当我注释掉这一行时:

//ptr = (struct sockaddr_in *) res->ai_addr;
Run Code Online (Sandbox Code Playgroud)

我会得到:

$ gcc ex4.c
ex4.c:30:9: error: expected expression
        struct sockaddr_in *sockAddrIn = (struct sockaddr_in *) res->ai_addr;
        ^
1 error generated.
Run Code Online (Sandbox Code Playgroud)

我错过了什么?

joh*_*boy 111

从技术上讲,switch语句中的每个case都是一个标签.出于某些模糊和旧的原因,您不能将变量声明作为标签后的第一行.通过评论分配

ptr = (struct sockaddr_in *) res->ai_addr;
Run Code Online (Sandbox Code Playgroud)

这条线

struct sockaddr_in *sockAddrIn = (struct sockaddr_in *) res->ai_addr;
Run Code Online (Sandbox Code Playgroud)

成为标签之后的第一行AF_INET:,正如我所说,在C中是非法的.

解决方案是将所有case语句包装在大括号中,如下所示:

#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <netdb.h>
#include <sys/types.h>
#include <sys/socket.h>
#include <arpa/inet.h>

int main (void) {

  struct addrinfo hints; 
  memset (&hints, 0, sizeof hints);

  hints.ai_family = AF_UNSPEC; 
  hints.ai_socktype = SOCK_DGRAM;  
  hints.ai_flags = AI_CANONNAME;   

  struct addrinfo *res;

  getaddrinfo ("example.com", "http", &hints, &res);
  printf ("Host: %s\n", "example.com");

  void *ptr;

  while (res != NULL) {
    printf("AI Family for current addrinfo: %i\n", res->ai_family);
    switch (res->ai_family) {
      case AF_INET:
      {
        ptr = (struct sockaddr_in *) res->ai_addr;
        struct sockaddr_in *sockAddrIn = (struct sockaddr_in *) res->ai_addr;
        break;
      }
    }
    res = res->ai_next;
  }
  return 0;
}
Run Code Online (Sandbox Code Playgroud)

无论如何,我认为这是更好的编码实践.

  • 尼斯.挑剔"每个案例......都是......一个标签声明".这就是原因:语句可以被标记但不能声明. (22认同)
  • @BlueMoon你是对的.当你可以[购买](http://www.bdsoft.com/tools/stlfilt.html)一个简化C++ stl错误消息的工具时,你知道信息密度非常低! (5认同)
  • @KorayTugay有时候编译器信息不能提供信息,因为我们希望它们是....有时候它们也非常有用......(咳嗽咳嗽C++ stl咳嗽). (4认同)
  • 或者只是在切换块之前移动声明 (3认同)
  • @immibis:在C++声明中是语句,这就是为什么在C++中你可以没有任何限制地标记声明.在C声明中不是语句,这就是为什么你不能标记它们.这是一个说明C和C++之间差异的例子:http://stackoverflow.com/a/19830820/187690 (3认同)

Mat*_*uez 15

作为已接受答案的补充,您可以在案例标签之前声明变量.

switch(a) {
    int b; //can't initialize variable here
    case 0:
    ...
}
Run Code Online (Sandbox Code Playgroud)

或者只使用一个空语句.