-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathHashMap.c
125 lines (108 loc) · 2.66 KB
/
HashMap.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
#include <stdlib.h>
#include <string.h>
#include "HashMap.h"
// We fix the number of hash buckets for simplicity.
#define N_BUCKETS 8
typedef struct Pair Pair;
struct Pair {
char *key;
void *value;
Pair *next; // Next item in a single-linked list.
};
struct HashMap {
Pair *buckets[N_BUCKETS]; // Linked lists of key-value pairs.
size_t size; // total number of entries in map.
};
static unsigned int get_hash(const char *key);
HashMap *hmap_new() {
HashMap *map = malloc(sizeof(HashMap));
if (!map)
return NULL;
memset(map, 0, sizeof(HashMap));
return map;
}
void hmap_free(HashMap *map) {
for (int h = 0; h < N_BUCKETS; ++h) {
for (Pair *p = map->buckets[h]; p;) {
Pair *q = p;
p = p->next;
free(q->key);
free(q);
}
}
free(map);
}
static Pair *hmap_find(HashMap *map, int h, const char *key) {
for (Pair *p = map->buckets[h]; p; p = p->next) {
if (strcmp(key, p->key) == 0)
return p;
}
return NULL;
}
void *hmap_get(HashMap *map, const char *key) {
int h = get_hash(key);
Pair *p = hmap_find(map, h, key);
if (p)
return p->value;
else
return NULL;
}
bool hmap_insert(HashMap *map, const char *key, void *value) {
if (!value)
return false;
int h = get_hash(key);
Pair *p = hmap_find(map, h, key);
if (p)
return false; // Already exists.
Pair *new_p = malloc(sizeof(Pair));
new_p->key = strdup(key);
new_p->value = value;
new_p->next = map->buckets[h];
map->buckets[h] = new_p;
map->size++;
return true;
}
bool hmap_remove(HashMap *map, const char *key) {
int h = get_hash(key);
Pair **pp = &(map->buckets[h]);
while (*pp) {
Pair *p = *pp;
if (strcmp(key, p->key) == 0) {
*pp = p->next;
free(p->key);
free(p);
map->size--;
return true;
}
pp = &(p->next);
}
return false;
}
size_t hmap_size(HashMap *map) {
return map->size;
}
HashMapIterator hmap_iterator(HashMap *map) {
HashMapIterator it = {0, map->buckets[0]};
return it;
}
bool
hmap_next(HashMap *map, HashMapIterator *it, const char **key, void **value) {
Pair *p = it->pair;
while (!p && it->bucket < N_BUCKETS - 1) {
p = map->buckets[++it->bucket];
}
if (!p)
return false;
*key = p->key;
*value = p->value;
it->pair = p->next;
return true;
}
static unsigned int get_hash(const char *key) {
unsigned int hash = 17;
while (*key) {
hash = (hash << 3) + hash + *key;
++key;
}
return hash % N_BUCKETS;
}