-
Notifications
You must be signed in to change notification settings - Fork 19
/
Copy pathCountingSort.c
203 lines (165 loc) · 6.24 KB
/
CountingSort.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
#include "../Headers/CountingSort.h"
#include "../../../System/Utils.h"
#include "../../../DataStructure/Tables/Headers/HashMap.h"
#include "../../../Unit Test/CuTest/CuTest.h"
/** This function takes an integer pointer,
* and then freeing it.
*
* Note: this function will be useful in the countingSortH function, because of the hash map.
*
* @param n the integer pointer
*/
void intFreeFunCountSort(void *n) {
free(n);
}
/** This function will take two integers pointers,
* then it will compare them.
*
* Note: this function will be useful in the countingSortH function, because of the hash map.
*
* @param n1 the first integer pointer
* @param n2 the second integer pointer
* @return it will return the result of the comparison, zero if they are equal, minus integer if the second bigger, and positive integer if the first bigger
*/
int intCmpCountSort(const void *n1, const void *n2) {
return *(int *) n1 - *(int *) n2;
}
/** This function takes an integer number pointer,
* then it will allocate a new integer space and copy the passed integer in the new space,
* and finally return the new integer pointer.
*
* @param num the number pointer
* @return it will return the new allocated pointer
*/
unsigned int *generateIntPointerCountSort(unsigned int *num) {
unsigned int *newInt = (unsigned int *) malloc(sizeof(unsigned int));
memcpy(newInt, num, sizeof(unsigned int));
return newInt;
}
/** This function will take an integer pointer as a parameter,
* then it will return the value of the integer.
*
* Note: this function will be useful to use in the hash map data structure.
*
* @param item the integer pointer
* @return it will return the value of the integer as the unique hash key
*/
int intHashFunCountSort(const void *item) {
return *(int *) item;
}
/** This function will sort an unsigned int array, using the counting sort algorithm.
* This function takes the range of the number that are existing in the array.
*
* Note: this function will use a fixed length array to count the numbers.
*
* Time Complexity: worst: O(n) , best: O (n).
*
* Space Complexity: O(k) and k is the range of the numbers.
*
* @param arr the array pointer
* @param length the length of the array
* @param rangeStart the start range number << ex: [2,50] then the start is 2 >>
* @param rangeEnd the end range number << ex: [2,50] then the end is 50 >>
*/
void countingSortA(unsigned int *arr, int length, unsigned int rangeStart, unsigned int rangeEnd) {
if (arr == NULL) {
#ifdef C_DATASTRUCTURES_ERRORSTESTSTRUCT_H
ERROR_TEST->errorCode = NULL_POINTER;
return;
#else
fprintf(stderr, NULL_POINTER_MESSAGE, "passed array", "counting sort");
exit(NULL_POINTER);
#endif
} else if (length < 0) {
#ifdef C_DATASTRUCTURES_ERRORSTESTSTRUCT_H
ERROR_TEST->errorCode = INVALID_ARG;
return;
#else
fprintf(stderr, INVALID_ARG_MESSAGE, "array length", "counting sort");
exit(INVALID_ARG);
#endif
} else if (rangeEnd < rangeStart) {
#ifdef C_DATASTRUCTURES_ERRORSTESTSTRUCT_H
ERROR_TEST->errorCode = INVALID_ARG;
return;
#else
fprintf(stderr, INVALID_ARG_MESSAGE, "start and end range", "counting sort");
exit(INVALID_ARG);
#endif
}
unsigned int countingArrLength = rangeEnd - rangeStart + 1;
unsigned int *countingArr = (unsigned int *) calloc(sizeof(unsigned int), countingArrLength);
for (int i = 0; i < length; i++)
countingArr[arr[i] - rangeStart]++;
int index = 0;
for (int i = 0; i < countingArrLength; i++) {
while (countingArr[i] != 0) {
arr[index++] = i + rangeStart;
countingArr[i]--;
}
}
}
/** This function will sort an unsigned int array, using the counting sort algorithm.
* This function takes the range of the number that are existing in the array.
*
* Note: this function will use a hash map to count the numbers.
*
* Time Complexity: worst: O(n) , best: O (n).
*
* Space Complexity: O(n) << because of the hash map>>.
*
* @param arr the array pointer
* @param length the length of the array
* @param rangeStart the start range number << ex: [2,50] then the start is 2 >>
* @param rangeEnd the end range number << ex: [2,50] then the end is 50 >>
*/
void countingSortH(unsigned int *arr, int length, unsigned int rangeStart, unsigned int rangeEnd) {
if (arr == NULL) {
#ifdef C_DATASTRUCTURES_ERRORSTESTSTRUCT_H
ERROR_TEST->errorCode = NULL_POINTER;
return;
#else
fprintf(stderr, NULL_POINTER_MESSAGE, "passed array", "counting sort");
exit(NULL_POINTER);
#endif
} else if (length < 0) {
#ifdef C_DATASTRUCTURES_ERRORSTESTSTRUCT_H
ERROR_TEST->errorCode = INVALID_ARG;
return;
#else
fprintf(stderr, INVALID_ARG_MESSAGE, "array length", "counting sort");
exit(INVALID_ARG);
#endif
} else if (rangeEnd < rangeStart) {
#ifdef C_DATASTRUCTURES_ERRORSTESTSTRUCT_H
ERROR_TEST->errorCode = INVALID_ARG;
return;
#else
fprintf(stderr, INVALID_ARG_MESSAGE, "start and end range", "counting sort");
exit(INVALID_ARG);
#endif
}
HashMap *countingMap = hashMapInitialization(intFreeFunCountSort, intFreeFunCountSort, intCmpCountSort,
intHashFunCountSort);
for (int i = 0; i < length; i++) {
unsigned int *newNumPointer = generateIntPointerCountSort(arr + i);
if (!hashMapContains(countingMap, newNumPointer)) {
unsigned int initialCount = 1;
unsigned int *countPointer = generateIntPointerCountSort(&initialCount);
hashMapInsert(countingMap, newNumPointer, countPointer);
} else {
unsigned int *currentCount = hashMapGet(countingMap, newNumPointer);
*currentCount += 1;
unsigned int *newCount = generateIntPointerCountSort(currentCount);
hashMapInsert(countingMap, newNumPointer, newCount);
}
}
for (unsigned int i = rangeStart, index = 0; i <= rangeEnd; i++) {
unsigned int *currentCount = (unsigned int *) hashMapGet(countingMap, &i);
while (currentCount != NULL && *currentCount > 0) {
arr[index++] = i;
*currentCount -= 1;
}
}
destroyHashMap(countingMap);
}