Tabla de picadillo

La versión actual de la página aún no ha sido revisada por colaboradores experimentados y puede diferir significativamente de la versión revisada el 12 de octubre de 2019; las comprobaciones requieren 9 ediciones .
Tabla de picadillo
Tipo de matriz asociativa
año de invención 1953
Complejidad en los símbolos O
Promedio Lo peor
Consumo de memoria En) En)
Búsqueda O(1) En)
Insertar O(1) En)
Eliminación O(1) En)
 Archivos multimedia en Wikimedia Commons

Una tabla hash  es una estructura de datos que implementa la interfaz de matriz asociativa , es decir, le permite almacenar pares (clave, valor) y realizar tres operaciones: la operación de agregar un nuevo par, la operación de búsqueda y la operación de eliminar un par por clave.

Introducción

Hay dos variantes principales de tablas hash: direccionamiento encadenado y abierto. La tabla hash contiene una matriz cuyos elementos son pares (una tabla hash de direcciones abiertas) o listas de pares (una tabla hash encadenada).

La ejecución de una operación en una tabla hash comienza con el cálculo de la función hash de la clave. El valor hash resultante actúa como un índice en la matriz . Luego, la operación que se está realizando (agregar, eliminar o buscar) se redirige al objeto, que se almacena en la celda correspondiente de la matriz .

La situación en la que se obtiene el mismo valor hash para diferentes claves se denomina colisión . Dichos eventos no son tan raros; por ejemplo, al insertar solo 23 elementos en una tabla hash de 365 celdas, la probabilidad de una colisión ya superará el 50% (si cada elemento puede caer en cualquier celda con la misma probabilidad): vea el cumpleaños paradoja _ Por lo tanto, el mecanismo de resolución de colisiones es una parte importante de cualquier tabla hash.

En algunos casos especiales, es posible evitar las colisiones por completo. Por ejemplo, si todas las claves de los elementos se conocen de antemano (o cambian muy raramente), entonces para ellos es posible encontrar alguna función hash perfecta que los distribuya entre las celdas de la tabla hash sin colisiones. Las tablas hash que utilizan estas funciones hash no necesitan un mecanismo de resolución de colisiones y se denominan tablas hash de direcciones directas .

El número de elementos almacenados dividido por el tamaño de la matriz (el número de valores hash posibles) se denomina factor de carga de la tabla hash y es un parámetro importante que determina el tiempo medio de ejecución de las operaciones.

Propiedades de la tabla hash

Una propiedad importante de las tablas hash es que, bajo algunos supuestos razonables, las tres operaciones (búsqueda, inserción, eliminación de elementos) se completan en promedio en el tiempo . Sin embargo, esto no garantiza que el tiempo de ejecución de una sola operación sea pequeño. Esto se debe al hecho de que cuando se alcanza un determinado valor del factor de relleno, es necesario reconstruir el índice de la tabla hash: aumentar el valor del tamaño de la matriz y volver a agregar todos los pares a la tabla hash vacía.

Resolución de colisiones

Hay varias formas de resolver las colisiones .

Método de la cadena

Cada celda de la matriz H es un puntero a una lista enlazada (cadena) de pares clave-valor correspondientes al mismo valor hash clave. Las colisiones simplemente dan como resultado cadenas que son más largas que un elemento.

Encontrar o eliminar un elemento requiere buscar en todos los elementos de la cadena correspondiente para encontrar un elemento con una clave determinada. Para agregar un elemento, debe agregar un elemento al final o al principio de la lista correspondiente, y si el factor de relleno se vuelve demasiado grande, aumente el tamaño de la matriz H y reconstruya la tabla.

Suponiendo que cada elemento puede caer en cualquier posición de la tabla H con la misma probabilidad e independientemente de dónde termine cualquier otro elemento, el tiempo promedio de la operación de búsqueda de elementos es Θ (1 + α ), donde α  es el factor de llenado de la tabla.

Direccionamiento abierto

La matriz H almacena los pares clave-valor. El algoritmo de inserción de elementos comprueba las celdas de la matriz H en algún orden hasta que se encuentra la primera celda libre, en la que se escribirá el nuevo elemento. Este orden se calcula sobre la marcha, lo que ahorra memoria para los punteros necesarios en las tablas hash encadenadas.

La secuencia en la que se buscan las celdas de la tabla hash se denomina secuencia de sonda . En el caso general depende solo de la clave del elemento, es decir, es una secuencia h 0 ( x ), h 1 ( x ), ..., h n  - 1 ( x ), donde x  es la clave del elemento , y h i ( x ) - funciones arbitrarias que asignan cada clave a una celda en la tabla hash. El primer elemento de la secuencia, por regla general, es igual al valor de alguna función hash de la clave, y el resto se calcula a partir de una de las siguientes formas. Para que los algoritmos de búsqueda funcionen correctamente, la secuencia de sondeos debe ser tal que todas las celdas de la tabla hash se escaneen exactamente una vez.

El algoritmo de búsqueda busca en las celdas de la tabla hash en el mismo orden que al insertar, hasta que se encuentra un elemento con la clave deseada o una celda libre (lo que significa que no hay ningún elemento en la tabla hash).

Eliminar elementos en tal esquema es algo difícil. Por lo general, hacen esto: configuran una bandera booleana para cada celda, marcando si un elemento en ella se ha eliminado o no. Luego, la eliminación de un elemento consiste en establecer esta bandera para la celda correspondiente de la tabla hash, pero al mismo tiempo es necesario modificar el procedimiento de búsqueda de un elemento existente para que considere ocupadas las celdas eliminadas, y el procedimiento por agregarlos para que los considere libres y restablezca el valor de la bandera cuando se agreguen.

Secuencias de muestra

Los siguientes son algunos tipos comunes de secuencias de muestra. Especifiquemos de inmediato que la numeración de elementos de la secuencia de muestras y celdas de la tabla hash es desde cero, y N  es el tamaño de la tabla hash (y, como se señaló anteriormente, también la longitud de la secuencia de muestras).


A continuación se muestra un código que demuestra doble hashing:

Implementación en C // ¡DICT_CELL_COUNT debe ser un número primo! #define DICT_CELL_COUNT 30011 char * szWordAr [ DICT_CELL_COUNT ]; sin firmar int uWordArSize = 0 ; #define WORD_IDX_BAD ((int sin firmar)-1) int sin firmar uWordIdxByHashAr [ DICT_CELL_COUNT ]; // necesita inicializar los elementos con el valor WORD_IDX_BAD #definir STRIDE_1 17 #definir STRIDE_2 13 // La función GetOrAddWordIdx( .. ) devuelve el índice de la palabra pcszWord en la matriz szWordAr. // Esto agrega la palabra pcszWord al diccionario szWordAr si no está allí. int sin firmar GetOrAddWordIdx ( const char * const pcszWord ) { int sin firmar uHash1 = 0 , uHash2 = 0 ; const carácter sin signo * cbtWordCharCur = ( const carácter sin signo * ) pcszWord ; // Calcula dos hashes de la palabra pcszWord: // uHash1 se encuentra en el rango [ 0 .. DICT_CELL_COUNT - 1 ] // uHash2 se encuentra en el rango [ 1 .. DICT_CELL_COUNT - 1 ] hacer { uHash1 *= STRIDE_1 ; uHash1 += ( STRIDE_2 * * cbtWordCharCur ); uHash2 *= STRIDE_2 ; uHash2 += ( STRIDE_1 * * cbtWordCharCur ); } while ( * ( cbtWordCharCur ++ ) ); // NB: incremento! // #1: cbtWordCharCur apunta al último carácter. '\0' en pcszWord, // se usará en #2 uHash1 %= DICT_CELL_COUNT ; uHash2 %= ( DICT_CELL_COUNT - 1 ); ++ uHash2 ; while ( uWordIdxByHashAr [ uHash1 ] != WORD_IDX_BAD ) { if ( ! strcmp ( pcszWord , szWordAr [ uWordIdxByHashAr [ uHash1 ] ] ) ) devuelve uWordIdxByHashAr [ uHash1 ]; uHash1 += uHash2 ; uHash1 %= DICT_CELL_COUNT ; } strcpy ( szPalabraAr [ uWordIdxByHashAr [ uHash1 ] = // NB: asignación! uWordArSize ] = // NB: tarea! ( char * ) malloc ( // longitud de pcszWord más 1: ( const char * ) cbtWordCharCur - // #2: use el valor cbtWordCharCur de #1 pcszWord ), pcszPalabra ); devuelve uWordArSize ++ ; // NB: incremento! } // int sin firmar GetOrAddWordIdx( const char* const pcszWord )

Véase también

Literatura

  • Kormen, T. , Leizerson, C. , Rivest, R. , Stein, K. Capítulo 11. Tablas hash. // Algoritmos: construcción y análisis = Introducción a los Algoritmos / Ed. I. V. Krasikova. - 2ª ed. - M. : Williams, 2005. - 1296 p. — ISBN 5-8459-0857-4 .