-
Notifications
You must be signed in to change notification settings - Fork 369
/
Copy pathMergeSortLinkedList.c
236 lines (187 loc) · 8.06 KB
/
MergeSortLinkedList.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
// =================================== Problem Statement ====================================
/*
Task: Sorting Algorithm for Linked List
Sol: The problem to sort the linked can be done via folowing methods
Method 1) One of the brute force solution is to convert the linked list into an array, sort the array using a standard sorting algorithm, and then rebuild the linked list using the sorted values. This approach may be simpler to implement but requires additional memory to store the array.
Time Complexity - O(n logn) , Space Complexity - O(n)
Method 2) Quick Sort can be applied to a linked list by selecting a pivot, partitioning the list into smaller and greater elements, and recursively sorting the partitions. The main challenge lies in rearranging the nodes during the partitioning step, as random access is not available. Despite this, Quick Sort remains an efficient sorting algorithm for linked lists with an average Time complexity of O(n log n) but in worst case O(n^2) and a Space complexity of O(log n).
Method 3) Merge Sort can be applied to a linked list by recursively dividing it into smaller sublists, sorting each sublist individually, and then merging the sorted sublists back together. The merging process takes advantage of the sorted order of the sublists, resulting in a sorted linked list. Merge Sort is efficient for linked lists, with a time complexity of O(n log n) and a space complexity of O(log n) due to the recursive nature of the algorithm.
Among the sorting algorithms suitable for linked lists, Merge Sort stands out as the most suitable choice. Merge Sort is a popularly known algorithm that follows the Divide and Conquer strategy. It can be implemented by recursively dividing the linked list, sorting each sublist, and merging them back together. However, even within Merge Sort, there are two approaches to consider:
Approach 1) The recursive approach using top-down merging.
- Recursively split the original list into two halves. The split continues until there is only one node in the linked list (Divide phase). To split the list into two halves, we find the middle of the linked list using the Fast and Slow pointer approach.
- Recursively sort each sublist and combine it into a single sorted list (Merge Phase).
Time Complexity: O(n logn) , Space Complexity: O(logn) for recursive stack space.
Approach 2) The iterative approach using bottom-up merging.
- Start with splitting the list into sublists of size 1.Each adjacent pair of sublists of size 1 is merged in sorted order. After the first iteration, we get the sorted lists of size 2.
- A similar process is repeated for a sublist of size 2.In this way, we iteratively split the list into sublists of size 1, 2, 4, 8, and so on until we reach n.
- To split the list into two sublists of given size beginning from start, we use two pointers: mid and end. Mid and end reference the start and end of the second linked list, respectively.
- Merge the lists in sorted order as discussed in Approach 1.
- As we iteratively split the list and merge, we have to keep track of the previous merged list using pointer tail and the next sublist to be sorted using pointer nextSubList.
Time Complexity: O(n logn) , Space Complexity: O(1) only reference pointer.
We are going to implemet both Approaches below
*/
// =================================== Code with C ==========================================
#include <stdio.h>
#include <stdlib.h>
// Node structure for linked list
struct Node
{
int data;
struct Node *next;
};
// Function to merge two sorted linked lists
struct Node *merge(struct Node *left, struct Node *right)
{
if (left == NULL)
return right;
if (right == NULL)
return left;
struct Node *result = NULL;
// Choose the smaller value as the head of the merged list
if (left->data <= right->data)
{
result = left;
result->next = merge(left->next, right);
}
else
{
result = right;
result->next = merge(left, right->next);
}
return result;
}
// Function to split the linked list into two halves
void splitList(struct Node *head, struct Node **left, struct Node **right)
{
struct Node *slow = head;
struct Node *fast = head->next;
// Move 'fast' two nodes ahead and 'slow' one node ahead
while (fast != NULL)
{
fast = fast->next;
if (fast != NULL)
{
slow = slow->next;
fast = fast->next;
}
}
*left = head;
*right = slow->next;
slow->next = NULL;
}
// Top-down merge sort on linked list
void mergeSortTopDown(struct Node **headRef)
{
struct Node *head = *headRef;
struct Node *left;
struct Node *right;
// Base case: if list is empty or contains only one node
if (head == NULL || head->next == NULL)
return;
// Split the list into two halves
splitList(head, &left, &right);
// Recursively sort the two halves
mergeSortTopDown(&left);
mergeSortTopDown(&right);
// Merge the sorted halves
*headRef = merge(left, right);
}
// Bottom-up merge sort on linked list
struct Node *mergeSortBottomUp(struct Node *head)
{
// Base case: if list is empty or contains only one node
if (head == NULL || head->next == NULL)
return head;
int listSize = 0;
struct Node *current = head;
// Count the number of nodes in the list
while (current != NULL)
{
listSize++;
current = current->next;
}
struct Node *dummy = (struct Node *)malloc(sizeof(struct Node));
dummy->next = head;
for (int blockSize = 1; blockSize < listSize; blockSize *= 2)
{
struct Node *prev = dummy;
current = dummy->next;
while (current != NULL)
{
// Split the list into two parts
struct Node *left = current;
struct Node *right;
for (int i = 1; i < blockSize && current->next != NULL; i++)
{
current = current->next;
}
right = current->next;
current->next = NULL;
current = right;
// Split the list again to get the next part
for (int i = 1; i < blockSize && current != NULL && current->next != NULL; i++)
{
current = current->next;
}
if (current != NULL)
{
struct Node *temp = current->next;
current->next = NULL;
current = temp;
}
// Merge the two parts
prev->next = merge(left, right);
while (prev->next != NULL)
prev = prev->next;
}
}
head = dummy->next;
free(dummy);
return head;
}
// Function to insert a new node at the beginning of the linked list
void insertNode(struct Node **headRef, int value)
{
struct Node *newNode = (struct Node *)malloc(sizeof(struct Node));
newNode->data = value;
newNode->next = *headRef;
*headRef = newNode;
}
// Function to print the linked list
void printList(struct Node *head)
{
struct Node *current = head;
while (current != NULL)
{
printf("%d ", current->data);
current = current->next;
}
printf("\n");
}
int main()
{
struct Node *headTopDown = NULL;
struct Node *headBottomUp = NULL;
// Insert nodes into the linked list
insertNode(&headTopDown, 5);
insertNode(&headTopDown, 2);
insertNode(&headTopDown, 8);
insertNode(&headTopDown, 1);
insertNode(&headTopDown, 9);
insertNode(&headBottomUp, 5);
insertNode(&headBottomUp, 2);
insertNode(&headBottomUp, 8);
insertNode(&headBottomUp, 1);
insertNode(&headBottomUp, 9);
printf("Original list: ");
printList(headTopDown);
// Merge sort using top-down approach
mergeSortTopDown(&headTopDown);
printf("Sorted list (Top-down): ");
printList(headTopDown);
// Merge sort using bottom-up approach
headBottomUp = mergeSortBottomUp(headBottomUp);
printf("Sorted list (Bottom-up): ");
printList(headBottomUp);
return 0;
}