-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathhash_table.c
306 lines (251 loc) · 8.74 KB
/
hash_table.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
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
/*
* TEMPLATE FROM: CS 11, C Track, lab 7
* http://users.cms.caltech.edu/~mvanier/CS11_C/labs/7/lab7.html
*
* PROGRAMMED BY : Jade Tran
* CLASS : CS1D
* SECTION : TTh 11:3 - 1:50
* FILE: hash_table.c
* Include the declaration of the hash table data structures
* and the function prototypes.
* ASSIGNMENT : A3 - Hash Table part 2
*/
#include <stdio.h>
#include <assert.h>
#include <stdlib.h>
#include <string.h>
#include <stdbool.h>
#include "hash_table.h"
/*** Hash function. ***/
/*Function used to hashing key which is used as index in hash_table
*Algorithm: "key" mod number of slots in hash_table
*Precondition: value being passed is valid (here is phone number)
*Postcondition: return "key"
*/
int hash(char *s)
{
int i = 0; /*Variable control loop*/
while(*s != '\0')
{
i = i + *s;
s++;
}
return (i % SLOTS); /*Return the result of i mod SLOTS*/
}
/*** Linked list utilities. ***/
/* Create a single node. */
/*Function used to create a new node
*Algorithm: create and dynamically allocate new node
*Precondition: value being passed are valid
*Postcondition: assign new_node->key = pointer variable type char being passed
* new_node->value = variable type int being passed
* new_node tail points to NULL
* return new node
*/
node *create_node(char *key, int value)
{
node *temp = (node *)malloc(sizeof(node)); /*Create pointer temp type and Dynamically allocate*/
temp->key = key;
temp->value = value;
temp->next = NULL; /*Point to NULL after creating a new node*/
return temp;
}
/* Look up the passed value in the hash table. */
/* This program must run in O(1) time */
/* return a pointer to the found or node or NULL */
/* if not found */
node *lookup(hash_table *ht, char *value)
{
node *head;
int i = 0;
/*Call hash function to find the slot*/
i = hash(value);
head = ht->slot[i];
if(head != NULL)
{
if(strcmp(head->key, value) == 0)
return head;
while(head->next != NULL)
{
if(strcmp(head->key, value) == 0)
return head;
head = head->next;
}
}
return NULL;
}
/* Delete will attempt to find the value in */
/* the hash table; if found will display a "found" message and delete */
/* the node, along with the corresponding value that is in memory. */
/* Otherwise will display a not found message. */
/* This function must run in O(1) time */
void delete(hash_table *ht, char *value)
{
node *found, *head, *temp;
int x = hash(value);
head = ht->slot[x];
/*Call lookup funtion*/
found = (node *)malloc(sizeof(node));
found = lookup(ht, value);
if(found == NULL)
{
printf("\n---Error! Can't find %s---\n", value);
printf("---Can't delete phone number---\n");
}
else
{
printf("---Found %s!---\n", value);
printf("---Error!---\n");
if(head->key == found->key)
{
printf("---Error1!---\n");
if(head->next != NULL)
head = found->next;
}
else
{
printf("---Error2!---\n");
while(head->key != found->key)
{
temp = found;
found = found->next;
}
temp->next = found->next;
}
free(found);
printf("---Successfully delete %s!---\n\n", value);
}
return;
}
/* Free all the nodes of a linked list. */
/*Function used to prevent memory leak from dynamically allocating variables
*Algorithm: using free()
*Precondition: node is not NULL; otherwise, get errors
*Postcondition: free() nodes in a linked list
*/
void free_list(node *list)
{
if (list == NULL) {
return;
}
if (list->next) {
free_list(list->next);
}
free(list);
list = NULL;
return;
}
/*** Hash table utilities. ***/
/* Create a new hash table. */
/*Function used to create hash table with size of SLOTS
*Algorithm: create and dynamically allocate hash table and number of slots (linked lists)
*Precondition: none
*Postcondition: create and initialize all nodes to be NULL then return hash table
*/
hash_table *create_hash_table()
{
int i; /*Variable control loop*/
hash_table *temp = (hash_table *)malloc(sizeof(hash_table)); /*Create pointer temp type hash_table*/
temp->slot = (node **)malloc(sizeof(node) * SLOTS); /*Dynamically allocate node temp->slot*/
for(i = 0; i < SLOTS; i++)
temp->slot[i] = NULL;
return temp;
}
/* Free a hash table. */
/*Function used to prevent memory leak from dynamically allocating variables
*Algorithm: using free()
*Precondition: hash_table is not NULL; otherwise, get errors
*Postcondition: free() linked lists and hash_table
*/
void free_hash_table(hash_table *ht)
{
int i = 0; /*Variable control loop*/
/*Loop through the linkedlist to free() memory*/
for(i = 0; i < SLOTS; i++)
{
free_list(ht->slot[i]);
}
free(ht);
return;
}
/* Get value */
/*Function used to determine whether the key after hash function is valid or invalid
*Algorithm: Look for a key in the hash table. Return 0 if not found.
* If it is found return the associated value.
*Precondition: hash_table is not NULL
*Postcondition: return 0 or return int
*/
int get_value(hash_table *ht, char *key)
{
/*Create a variable to store int returned from hash function*/
int value = hash(key);
if(ht->slot[value] == NULL)
return 0;
else
return value;
}
/* Set value */
/*Function used to add new node to hash table and set its value
*Algorithm: Set the value stored at a key. If the key is not in the table,
* create a new node and set the value to 'value'. Note that this
* function alters the hash table that was passed to it.
*Precondition: hash_table is already created
*/
void set_value(hash_table *ht, char *key, int value)
{
node *temp = NULL; /*Create pointer temp type node and initialize to NULL*/
node *head; /*Create pointer head type node*/
int x = 0; /*Variable x type int and initialize to 0*/
x = hash(key); /*Call hash function and store to variable x*/
temp = create_node(key, x); /*Call create node function and store node in variable temp*/
/*If the key is not in the table, create a new node consider 2 cases*/
if(value == 1) /*Case 1: no key in a linked list*/
ht->slot[x] = temp;
else /*Case 2: there is/are key(s) in a linked list*/
{
head = ht->slot[x]; /*traverse the list*/
while(head->next != NULL)
{
head = head->next;
}
head->next = temp;
num_coll = num_coll + 1; /*Count number of collisions*/
}
return;
}
/* Print out the contents of the hash table as key/value pairs. */
/*
*Precondition: hash_table and its slots are created
*Postcondition: if slot (linked list) is NULL, print in empty format
* else, print all the nodes with its "key"
*/
void print_hash_table(hash_table *ht)
{
int i = 0; /*Variable control loop*/
int x; /*Use to print nodes in linked list*/
node *head; /*Create pointer head type node*/
printf("\tValue\tKey\n\n");
for(i = 0; i < SLOTS; i++)
{
if(ht->slot[i] != NULL)
{
head = ht->slot[i];
do
{
if(x == 1)
{
printf("\t\t%s\n", ht->slot[i]->key);
}
else
{
printf("\t%i\t%s\n", i, ht->slot[i]->key);
x = 1;
}
head = head->next;
}while(head!=NULL);
}
else /*Empty format - nothing in the linked list*/
printf("\t%i\t----\n",i);
}
return;
}