Why hash tables are incredibly useful

When you want to look for a value in an array, looping over it is simple and fast, but it has a time complexity of O(n). This means that as the array gets really big, it can become slow to find an element in the array.

Hash tables are incredibly useful for turning code that does an O(n) iteration over an entire array, into an O(1) operation that (on average) instantly finds the element, no matter the size of the array.

It is important to keep in mind that iterating an array is most of the time going to be much faster in practice, since it is so simple. For this reason, and because the array version will be easier to read, I recommend only upgrading arrays to hash tables in C once a performance profiler indicates it to be necessary.

High-level explanation of my favorite implementation

This hash table requires you to have an array of items that you want to hash. main.c contains a persons array of person structs. Each person has a name and an age.

The pigeonhole principle tells us that if we have 3 persons, and we’re trying to fit them into 2 buckets, at least one of the buckets must contain more than one person.

In order to fit more than one person in each bucket, each bucket is a linked list.

Usually linked lists are implemented by having every node be malloc()ed separately, but main.c manages to do it with just has a single, global buckets and chains array.

I made these arrays are global in order to make the code easier to read, but you could of course just pass them as arguments to the functions.

Examples

Let’s assume that the bucket count is 2 for a moment.

main() then pushes into the persons array in this order:

  1. trez, with an age of 7
  2. john, with an age of 42
  3. carl, with an age of 69

The program hashes the person structs based on their names, so let’s use an imaginary hash() function:

  1. hash("trez") returns 42
  2. hash("john") returns 69
  3. hash("carl") returns 38

Here’s how those hashes are used to give buckets and chains their values, where nbucket is the number of buckets, and -1 is the sentinel value.

Bucket[i] has the value of the last entry that has bucket_index equal to i

 i  buckets[i]  name of first person in chain
--  ----------  -----------------------------
 0  2           carl
 1  1           john

One asterisk * indicates the start of a chain

    name =          | hash =      bucket_index =  chain =
 i  persons[i].name | hash(name)  hash % nbucket  chains[i]
--  --------------- | ----------  --------------  ---------
 0  trez            |  42         0               -1 <-\
 1  john            |  69         1 *             -1   |
 2  carl            |  38         0 *              0 --/

Let’s verify this table, using the algorithm get_person() in main.c uses:

  1. john:
hash("john") = 69
chain starts at buckets[69 % 2] = buckets[1] = 1

persons[1].name (= "john") == "john"? yes => "john" found at persons[1]
  1. bob:
hash("bob") = 420
chain starts at buckets[420 % 2] = buckets[0] = 2

persons[2].name (= "carl") == "bob"? no => chain continues at i=0
persons[0].name (= "trez") == "bob"? no => chain continues at i=-1
i=-1 is the sentinel index, so "bob" is not in the hash table

Implementation

The elf_hash() function here can be swapped out for any other hashing function, and can be reused for all of your hash tables. See the GitHub repository for more information. You can try it on Compiler Explorer here.

#include <stdint.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>

#define MAX_PERSONS 420420

struct person {
	char *name;
	uint32_t age;
};

static struct person persons[MAX_PERSONS];
static size_t persons_size;

static uint32_t buckets[MAX_PERSONS];
static uint32_t chains[MAX_PERSONS];

// From https://sourceware.org/git/?p=binutils-gdb.git;a=blob;f=bfd/elf.c#l193
static uint32_t elf_hash(const char *namearg) {
	uint32_t h = 0;
	for (const unsigned char *name = (const unsigned char *) namearg; *name; name++) {
		h = (h << 4) + *name;
		h ^= (h >> 24) & 0xf0;
	}
	return h & 0x0fffffff;
}

static struct person *get_person(char *name) {
	uint32_t i = buckets[elf_hash(name) % persons_size];

	while (1) {
		if (i == UINT32_MAX) {
			return NULL;
		}

		if (strcmp(name, persons[i].name) == 0) {
			break;
		}

		i = chains[i];
	}

	return persons + i;
}

static void hash_persons(void) {
	memset(buckets, UINT32_MAX, persons_size * sizeof(uint32_t));

	for (size_t i = 0; i < persons_size; i++) {
		uint32_t bucket_index = elf_hash(persons[i].name) % persons_size;

		chains[i] = buckets[bucket_index];

		buckets[bucket_index] = i;
	}
}

static void push_person(char *name, uint32_t age) {
	if (persons_size >= MAX_PERSONS) {
		fprintf(stderr, "There are more than %d persons, exceeding MAX_PERSONS\n", MAX_PERSONS);
		exit(EXIT_FAILURE);
	}
	persons[persons_size++] = (struct person){ .name = name, .age = age };
}

int main(void) {
	push_person("trez", 7);
	push_person("john", 42);
	push_person("carl", 69);

	hash_persons();

	printf("%d\n", get_person("trez")->age); // Prints "7"
	printf("%d\n", get_person("john")->age); // Prints "42"
	printf("%d\n", get_person("carl")->age); // Prints "69"
	printf("%p\n", (void *)get_person("bob")); // Prints "(nil)"
}

Profiling this hash table against an array

Hash tables are tricky to profile, since you would ideally test countless combinations of insertions, deletions, and searching.

In this case however we’re assuming that the hash table is only built once the entire array is never going to be modified again, so profile.c just compares how long it takes to figure out if a random element is present in a hash table and a regular array:

Screenshot of profiling

The timings here show that a hash table takes close to zero time no matter how many elements there are, whereas an array can start taking a ton of time.

I encourage you to scrutinize my tests, but my important takeaway here is that given enough memory, a hash table will always take pretty much zero time, whereas time_array starts taking a noticeable amount of time when its search limit argument is around 1000.

If you keep in mind however that ROUNDS is causing every search to be repeated 10000 times, the correct conclusion here is that searching an array with 100k elements only took around 2.3 milliseconds (so not 23.86 seconds).

Credits

My explanation is based on flapenguin’s great blog post called ELF: symbol lookup via DT_HASH.

The hash table implementation is taken from the code Linux uses to perform symbol lookup in ELF files. Its Linux Foundation documentation is here.