2009-04-13 5 views
39

Non riesco a usare boost: hash perché devo stare con C e non posso usare C++.Una funzione hash minima per C?

Ma, ho bisogno di hash un numero elevato (da 10K a 100k) di stringhe di token (da 5 a 40 byte di lunghezza) in modo che la ricerca all'interno di quelle sia la più veloce.

MD5, SHA1 o qualsiasi funzione di hash lunga sembra troppo pesante per un'operazione semplice, non sto facendo crittografia. Inoltre, vi è il costo di archiviazione e di elaborazione.

Quindi la mia domanda:

  1. Quale potrebbe essere il più semplice algoritmo di hash che garantirà la prevenzione collisioni nei casi più pratici.

  2. Quanti bit utilizzare per il valore hash? Sto sviluppando per sistemi a 32 bit. L'algoritmo hash in Perl/Python usa anche gli hash a 32 bit? O devo saltare a 64?

  3. Riguardo all'implementazione delle tabelle hash nei linguaggi di scripting comuni: il controllo dell'implementazione per le collisioni o posso evitare del tutto quella parte?

+23

la seguente pagina ha diverse implementazioni di funzioni hash uso generale implementati in C (e molte altre lingue): http://partow.net/ programmazione/hashfunctions/index.html –

+0

Hai considerato l'utilizzo di GLIB? https://developer.gnome.org/glib/2.46/glib-Hash-Tables.html –

risposta

23

È possibile trovare una buona (e veloce) funzione di hash, e una interessante lettura, a http://www.azillionmonkeys.com/qed/hash.html

L'unica volta in cui non è necessario verificare la presenza di collisioni, è se si utilizza un hash perfetto: una tabella di ricerca vecchio stile, come gperf.

+3

Suggerirei di guardare a quello che mancava all'analisi di Hsieh: MurmurHash2. http://en.wikipedia.org/wiki/MurmurHash –

7

Una funzione di hash generale per hash table lookup. Specifica NON utilizzare per scopi di crittografia, ma dal momento che hai specificato di non avere alcun intento, dovresti essere ok.

E 'incluso è A Survey di funzioni hash di provare

11
  1. Here è una bella panoramica delle più importanti funzioni di hash noti.

  2. 32 bit dovrebbe funzionare bene.

  3. È sempre necessario verificare la presenza di collisioni, a meno che non si vuole scrivere una tabella hash divertente :)

+0

Non è necessario verificare la presenza di collisioni se non si cura particolarmente della risposta che si ottiene. Il vantaggio è che non è necessario memorizzare la chiave originale nella tabella hash in modo da poter risparmiare molto spazio. –

+2

Beh, un tale comportamento non deterministico è ciò che intendevo per 'divertente'. – arul

2

Prova Adler32 per stringhe lunghe o Murmur2 per stringhe corte.

+3

Adler32 non è affatto un ottimo hash. In realtà, è anche peggio di CRC-32, come un hash. Murmur2, d'altra parte, è un hash molto veloce con un'eccellente distribuzione e comportamento nel caso peggiore, quindi non c'è motivo di limitare il suo uso alle stringhe corte. Non capisco davvero le basi del tuo consiglio. –

4

Se si utilizza un sistema posix allo stesso modo e si applica semplicemente C, vorrei semplicemente utilizzare ciò che il sistema ha già da offrire. man 3 hcreate ti offre tutti i dettagli oppure puoi trovare una versione online qui http://linux.die.net/man/3/hcreate

1

xxhash è un'opzione abbastanza veloce e facile. Un semplice codice userebbe XXH32 funzione:

unsigned int XXH32 (const void* input, int len, unsigned int seed); 

Si tratta di 32 bit di hash.Dal momento che è lenint, per i dati più grandi più di 2^31-1 byte utilizzano questi:

void*   XXH32_init (unsigned int seed); 
XXH_errorcode XXH32_update (void* state, const void* input, int len); 
unsigned int XXH32_digest (void* state);