Сбой коммутатора после комментирования неиспользуемой строки

Вот мой код:

#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;
}

который компилируется отлично.

Однако, когда я прокомментирую эту строку:

//ptr = (struct sockaddr_in *) res->ai_addr;

Я получу:

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

Что мне не хватает?

Ответ 1

Каждый случай в инструкции switch, технически говоря, является меткой. Для некоторых неясных и старых причин вам не разрешено иметь объявление переменной в качестве первой строки после метки. Комментируя назначение

ptr = (struct sockaddr_in *) res->ai_addr;

строка

struct sockaddr_in *sockAddrIn = (struct sockaddr_in *) res->ai_addr;

становится первой строкой после метки AF_INET:, которая, как я уже сказал, является незаконной в 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;
}

В любом случае, я думаю, что это лучше практика кодирования.

Ответ 2

В качестве дополнения к принятому ответу вы можете объявить свои переменные перед метками case.

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

Или просто используйте пустой оператор.