-
Notifications
You must be signed in to change notification settings - Fork 7
/
Copy pathmap.c
140 lines (124 loc) · 5.62 KB
/
map.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
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
/* *****************************************************************************
Hash Maps
Easily define key-value String Hash Map, also sometimes called a "dictionary",
using different smart defaults for short keys `FIO_MAP_KEY_KSTR` vs longer keys
(or when expecting a sparsely populated map) `FIO_MAP_KEY_BSTR`.
***************************************************************************** */
/* *****************************************************************************
(2) Hash Map: String Dictionary (automatic setup)
***************************************************************************** */
/* Set the properties for the key-value Unordered Map type called `dict_s` */
#define FIO_UMAP_NAME dict
#define FIO_MAP_KEY_KSTR /* pre-defined macro for using fio_keystr_s keys. */
#define FIO_MAP_VALUE_BSTR /* pre-defined macro for using String values. */
#define FIO_MAP_HASH_FN(str) \
fio_risky_hash(str.buf, str.len, (uint64_t)&fio_risky_hash)
#include "fio-stl/include.h" /* or "fio-stl.h" */
void easy_dict_example(void) {
dict_s dictionary = FIO_MAP_INIT;
/* insertion using dict_set */
dict_set(&dictionary,
FIO_STR_INFO1("Hello"),
FIO_STR_INFO1("Hello World!"),
NULL);
dict_set(&dictionary,
FIO_STR_INFO1("42"),
FIO_STR_INFO1("Meaning of life..."),
NULL);
/* access using dict_get */
printf("- Printing each key: value\n");
printf("Hello: %s\n", dict_get(&dictionary, FIO_STR_INFO1("Hello")).buf);
printf("42: %s\n", dict_get(&dictionary, FIO_STR_INFO1("42")).buf);
/* update using dict_set */
printf("- Updating value for 42 and reprinting\n");
dict_set(&dictionary,
FIO_STR_INFO1("42"),
FIO_STR_INFO1("What was the question?"),
NULL);
/* map iteration - this is an unordered map, order is incidental. */
FIO_MAP_EACH(dict, &dictionary, i) {
printf("%-8s: %s\n", i.key.buf, i.value.buf);
}
/* removal using dict_remove */
dict_remove(&dictionary, FIO_STR_INFO1("42"), NULL);
/* Since the "42" key was removed, its `buf` value will point to NULL. */
printf("Did we remove 42 ... ? - %s\n",
dict_get(&dictionary, FIO_STR_INFO1("42")).buf
? dict_get(&dictionary, FIO_STR_INFO1("42")).buf
: "removed");
/* Cleanup. */
dict_destroy(&dictionary);
}
/* *****************************************************************************
(2) Hash Map: String Dictionary (manual setup)
***************************************************************************** */
/* Create a binary safe String type for Strings that aren't mutated often */
#define FIO_STR_SMALL str
#include "fio-stl/include.h" /* or "fio-stl.h" */
/* Defines a key-value Unordered Map type called `dictionary_s` */
#define FIO_MAP_NAME dictionary
#define FIO_MAP_ORDERED 0
#define FIO_MAP_VALUE str_s
#define FIO_MAP_VALUE_COPY(dest, src) str_init_copy2(&(dest), &(src))
#define FIO_MAP_VALUE_DESTROY(k) str_destroy(&k)
#define FIO_MAP_VALUE_CMP(a, b) str_is_eq(&(a), &(b))
#define FIO_MAP_KEY FIO_MAP_VALUE
#define FIO_MAP_KEY_COPY FIO_MAP_VALUE_COPY
#define FIO_MAP_KEY_DESTROY FIO_MAP_VALUE_DESTROY
#define FIO_MAP_KEY_CMP FIO_MAP_VALUE_CMP
#include "fio-stl/include.h" /* or "fio-stl.h" */
/** set helper for consistent hash values */
FIO_IFUNC str_s dictionary_set2(dictionary_s *m, str_s key, str_s obj) {
return dictionary_set(m, str_hash(&key, (uint64_t)m), key, obj, NULL);
}
/** get helper for consistent hash values */
FIO_IFUNC str_s *dictionary_get2(dictionary_s *m, str_s key) {
return &(dictionary_get_ptr(m, str_hash(&key, (uint64_t)m), key)->value);
}
void dictionary_example(void) {
dictionary_s dictionary = FIO_MAP_INIT;
str_s key, val;
str_init_const(&key, "hello", 5);
str_init_const(&val, "Hello World!", 12);
dictionary_set2(&dictionary, key, val);
printf("%s\n", str_ptr(dictionary_get2(&dictionary, key)));
dictionary_destroy(&dictionary);
}
/* *****************************************************************************
(3) Hash Map: Strings to Numbers
***************************************************************************** */
/* map words to numbers. */
#define FIO_UMAP_NAME umap
#define FIO_MAP_KEY_KSTR
#define FIO_MAP_VALUE uintptr_t
#define FIO_MAP_HASH_FN(k) \
fio_risky_hash((k).buf, (k).len, (uint64_t)(uintptr_t)&umap_destroy)
#include "fio-stl/include.h" /* or "fio-stl.h" */
/* example adding strings to map and printing data. */
void map_keystr_example(void) {
umap_s map = FIO_MAP_INIT;
/* FIO_KEYSTR_CONST prevents copying of longer constant strings */
umap_set(&map, FIO_STR_INFO3("One", 3, FIO_KEYSTR_CONST), 1, NULL);
umap_set(&map, FIO_STR_INFO3("Two", 3, FIO_KEYSTR_CONST), 2, NULL);
umap_set(&map, FIO_STR_INFO3("Three", 5, FIO_KEYSTR_CONST), 3, NULL);
FIO_MAP_EACH(umap, &map, pos) {
uintptr_t value = pos.value;
printf("%.*s: %llu\n",
(int)pos.key.len,
pos.key.buf,
(unsigned long long)value);
}
umap_destroy(&map);
}
/* *****************************************************************************
main
***************************************************************************** */
int main(void) {
printf("=====================================\n");
easy_dict_example();
printf("=====================================\n");
dictionary_example();
printf("=====================================\n");
map_keystr_example();
printf("=====================================\n");
}