2010-02-22 6 views
7

Sto avendo un bizzarro problema con C++ in cui il lungo tipo di dati sta straripando molto prima di quanto dovrebbe. Quello che sto facendo (con successo finora) è che gli interi si comportano come float, quindi l'intervallo [-32767,32767] è mappato a [-1.0,1.0]. Dove si imbatte è con argomentazioni più grandi che rappresentano galleggia maggiore di 1,0:C++ traboccante a lungo prima del tempo

inline long times(long a, long b) { 
    printf("a=%ld b=%ld ",a,b); 
    a *= b; 
    printf("a*b=%ld ",a); 
    a /= 32767l; 
    printf("a*b/32767=%ld\n",a); 
    return a; 
} 

int main(void) { 
    printf("%ld\n",times(98301l,32767l)); 
} 

quello che ottengo come output è:

a=98301 b=32767 a*b=-1073938429 a*b/32767=-32775 
-32775 

così volte (98301,32767) è analogo al 3.0 * 1.0. Questo codice funziona perfettamente quando gli argomenti per i tempi sono inferiori a 32767 (1.0), ma nessuno dei passaggi intermedi con gli argomenti precedenti deve superare i 64 bit di lunghezza.

Qualche idea?

+0

Puoi accettare la risposta quindi, rende le persone più disponibili ad aiutarti in futuro. –

risposta

9

long non è necessariamente 64 bit. prova invece "long long".

+0

Funziona! Grazie a tutti coloro che hanno suggerito questo, che resistenza. Mi fa apprezzare molto di più Java ... – rhodri

+0

@rhodri: long long non è nemmeno necessariamente 64 bit, tra l'altro (su alcune implementazioni). Controlla Boost Integer o controlla se il tuo compilatore ha stdint.h. – GManNickG

+1

@GMan: corretto. Tuttavia, lo standard C garantisce che 'long long' ha * un intervallo sufficiente * per un intero a 64 bit, e di solito è abbastanza ... – sleske

2

Probabilmente avete i long a 32 bit. Prova a utilizzare long long invece.

98301 * 32767 = 3221028867, mentre una lunga overflow 32 bit a 2147483648

4

Il tipo long non è necessariamente 64 bit. Se sei sull'architettura a 32 bit (almeno su MS Visual C++), il tipo long è a 32 bit. Controllalo con sizeof (long). C'è anche il tipo di dati long long che può aiutare.

2

Lo standard C garantisce solo che long abbia almeno 32 bit (che è in realtà il caso della maggior parte delle piattaforme a 32 bit).

Se è necessario 64 bit, utilizzare long long. È garantito per contenere almeno 64 bit.