2013-08-19 16 views
5

Sto cercando di rilevare se un valore di tipo intero famiglia (char, unsigned char, short, unsigned short, int, ...) è un numero negativo in C. Se possibile con una macro che può essere compilato con qualsiasi compilatore C conforme (quindi, gcc -tricks consentito) e senza avviso!Come rilevare la firma con una macro?

Dopo qualche tempo sono venuto con il seguente:

#define ISNEG(X) ((X) && (X-1) && ((X <= 0) && (~X >= 0))) 

ho provato con i seguenti esempi:

void 
display_result(int arg, int result) 
{ 
    printf("ISNEG(%d) is %stive\n", arg, (result ? "nega" : "posi")); 
} 

void 
display_uresult(unsigned int arg, int result) 
{ 
    printf("ISNEG(%u) is %stive\n", arg, (result ? "nega" : "posi")); 
} 

int main() 
{ 
    short shrt = 5; 
    short nshrt = -5; 
    unsigned short ushrt = 5; 

    display_result(shrt, ISNEG(shrt)); 
    display_result(nshrt, ISNEG(nshrt)); 
    display_uresult(ushrt, ISNEG(ushrt)); 

    int ni = -5; 
    int i = 5; 
    int zero = 0; 

    display_result(ni, ISNEG(ni)); 
    display_result(i, ISNEG(i)); 
    display_result(zero, ISNEG(zero)); 
    display_result(~zero, ISNEG(~zero)); // wrong 

    unsigned int uzero = 0; 
    unsigned int ui = 5; 

    display_uresult(uzero, ISNEG(uzero)); 
    display_uresult(~uzero, ISNEG(~uzero)); 
    display_uresult(ui, ISNEG(ui)); 

    long int li = -5; 
    unsigned long int uli = 5; 

    display_result(li, ISNEG(li)); 
    display_uresult(uli, ISNEG(uli)); 

    long long int lli = -5; 
    unsigned long long int ulli = 5; 

    display_result(lli, ISNEG(lli)); 
    display_uresult(ulli, ISNEG(ulli)); 

    return EXIT_SUCCESS; 
} 

E, il risultato è:

ISNEG(5) is positive 
ISNEG(-5) is negative 
ISNEG(5) is positive 
ISNEG(-5) is negative 
ISNEG(5) is positive 
ISNEG(0) is positive 
ISNEG(-1) is negative 
ISNEG(0) is positive 
ISNEG(4294967295) is positive 
ISNEG(5) is positive 
ISNEG(-5) is negative 
ISNEG(5) is positive 
ISNEG(-5) is negative 
ISNEG(5) is positive 

E funziona abbastanza bello ma il problema è che, quando compilato con tutti gli avvertimenti (-Wall -Wextra), ottengo il seguito messaggi ing:

signedness.c: In function ‘main’: 
signedness.c:27:3: warning: promoted ~unsigned is always non-zero [-Wsign-compare] 
    display_uresult(ushrt, ISNEG(ushrt)); 
^
signedness.c:4:49: warning: comparison of unsigned expression >= 0 is always true [-Wtype-limits] 
#define ISNEG(X) ((X) && (X-1) && ((X <= 0) && (~X >= 0))) 
               ^
signedness.c:41:26: note: in expansion of macro ‘ISNEG’ 
    display_uresult(uzero, ISNEG(uzero)); 
         ^
signedness.c:4:49: warning: comparison of unsigned expression >= 0 is always true [-Wtype-limits] 
#define ISNEG(X) ((X) && (X-1) && ((X <= 0) && (~X >= 0))) 
               ^
signedness.c:42:27: note: in expansion of macro ‘ISNEG’ 
    display_uresult(~uzero, ISNEG(~uzero)); 
         ^
signedness.c:4:49: warning: comparison of unsigned expression >= 0 is always true [-Wtype-limits] 
#define ISNEG(X) ((X) && (X-1) && ((X <= 0) && (~X >= 0))) 
               ^
signedness.c:43:23: note: in expansion of macro ‘ISNEG’ 
    display_uresult(ui, ISNEG(ui)); 
        ^
signedness.c:4:49: warning: comparison of unsigned expression >= 0 is always true [-Wtype-limits] 
#define ISNEG(X) ((X) && (X-1) && ((X <= 0) && (~X >= 0))) 
               ^
signedness.c:49:24: note: in expansion of macro ‘ISNEG’ 
    display_uresult(uli, ISNEG(uli)); 
         ^
signedness.c:4:49: warning: comparison of unsigned expression >= 0 is always true [-Wtype-limits] 
#define ISNEG(X) ((X) && (X-1) && ((X <= 0) && (~X >= 0))) 
              ^
signedness.c:55:25: note: in expansion of macro ‘ISNEG’ 
    display_uresult(ulli, ISNEG(ulli)); 
         ^

Quindi, le mie domande sono:

  1. C'è un modo migliore per rilevare che noi abbiamo una variabile negativa tra tutti i possibili tipi interi del linguaggio C?

  2. Come sbarazzarsi di tutti questi avvisi senza disattivarlo (e senza utilizzare trucchi GCC)?

+0

possibile duplicato di [Macro per verificare se un determinato tipo intero è Firmato o Non firmato in C?] (Http://stackoverflow.com/questions/7962855/macro-to-test-whether-a-given- intero-type-is-signed-or-unsigned-in-c) – WhozCraig

+0

@WhozCraig: non sto parlando del tipo di variabile, ma del valore della variabile. Inoltre, data una variabile, C non ha introspezione, quindi non c'è modo di ottenere il tipo di variabile 'a priori' (eccetto se si sta usando un'estensione non standard). – perror

+0

Cosa c'è di sbagliato con '(X <0)'? È l'avviso del compilatore per i tipi non firmati? – jxh

risposta

5

Questa definizione sembra funzionare per me senza eventuali avvisi generati:

#define ISNEG(X) (!((X) > 0) && ((X) != 0)) 

Ho usato il vostro casi di test con questa macro su IDEONE.

Se siete su un sistema che supporta la funzione _Generic selezione C.11, allora si può fare qualcosa di simile (questo è una specie di semplice mente, io sono sicuro che potrebbe essere semplificato sfruttando regole promozione integrale):

#define ISNEG(X) \ 
    _Generic((X), \     
      char: !((X) > 0) && (X) != 0, \ 
      signed char: (X) < 0, \ 
      short: (X) < 0, \ 
      int: (X) < 0, \ 
      long: (X) < 0, \ 
      long long: (X) < 0, \ 
      float: (X) < 0, \ 
      double: (X) < 0, \ 
      long double: (X) < 0, \ 
      default: 0) 
+0

Mi piace il tuo #define ISNEG (X) (! ((X)> 0) && ((X)! = 0)) '. È ciò che io chiamo semplice e * elegante *! Grazie ! :-) – perror

+0

Attenzione a non creare codice con effetti collaterali! – gsk

+0

@gsk: Sì, è una preoccupazione per qualsiasi macro in cui l'argomento viene espanso più di una volta. – jxh

1

per me, questo ha funzionato bene:

definire ISNEG (X) ((X) < ((int) 0))

con questo risultato (http://ideone.com/9SIoHQ)

ISNEG(5) is positive 
ISNEG(-5) is negative 
ISNEG(5) is positive 
ISNEG(-5) is negative 
ISNEG(5) is positive 
ISNEG(0) is positive 
ISNEG(-1) is negative 
ISNEG(0) is positive 
ISNEG(4294967295) is positive 
ISNEG(5) is positive 
ISNEG(-5) is negative 
ISNEG(5) is positive 
ISNEG(-5) is negative 
ISNEG(5) is positive 
+0

Quando si utilizza 'gcc -funsigned-char -W', quindi usando questa macro su una variabile' char', ottengo: 'warning: il confronto è sempre falso a causa della gamma limitata di tipi di dati ' – jxh

Problemi correlati