KEMBAR78
11 Algorithms | PDF | Time Complexity | Theoretical Computer Science
0% found this document useful (0 votes)
18 views34 pages

11 Algorithms

Copyright
© © All Rights Reserved
We take content rights seriously. If you suspect this is your content, claim it here.
Available Formats
Download as DOCX, PDF, TXT or read online on Scribd
0% found this document useful (0 votes)
18 views34 pages

11 Algorithms

Copyright
© © All Rights Reserved
We take content rights seriously. If you suspect this is your content, claim it here.
Available Formats
Download as DOCX, PDF, TXT or read online on Scribd
You are on page 1/ 34

Algorithms Presentation

Bubble Sort
Idea:
 The bubble sort algorithm is a brute-force sorting algorithm.
 The algorithm iterates through the array, comparing and swapping
adjacent elements if they are not in the correct order.
Pseudo code:

1 Function BubbleSort(arr)
2 n ← length(arr)
3 for i ← 0 to n - 2 do
4 for j ← 0 to n - i - 2 do
5 if arr[j] > arr[j + 1] then
6 swap(arr[j], arr[j + 1])
7 end
8 end
9 end
1 end
0

Illustration:

Step Array Annotations

0 {5,3,2,1}

1 {3,5,2,1} swap (5, 3)


2 {3,2,5,1} swap (5, 2)
3 {3,2,1,5} swap (5, 1)
4 {2,3,1,5} swap (3, 2)
5 {2,1,3,5} swap (3, 1)
6 {1,2,3,5} swap (2, 1)

Complexity:
Space:
The algorithm does not requires extra arrays, so space complexity is
always Θ(1) for all cases.
Time:

Algorithms Best-case Worst-case Stable

Bubble Sort O(n) O(n2) Yes

 Base operator is comparison in line 5


 The number of times the basic operation is executed depends on n
and the nature of input.
Worst-case
 Sum of number of times the basic operation for the worst-case is:
n −2 n −i−2 n−2
n(n−1)
∑¿ ∑ 1 }=∑ ( n−i−1 )=
2
¿
i=0 i=0 i=0
 Order of growth: O(n2)
Best-case
 In the best-case, the input array is already sorted in ascending
or descending order.
 We can use an additional flag to track whether any swaps occurred
during an iteration. If no swaps occurred, it means the array is
already sorted, and we can stop the algorithm. This improvement
will make the best-case time complexity of Bubble Sort to be
O(n).
Optimize: Bubble sort with flag
 The algorithm is similar to regular Bubble Sort, but it adds a
flag to keep track of whether any swaps occurred during the last
iteration.
 If no swaps occurred, it means the array is already sorted, so
the algorithm can stop.

1 Function BubbleSort_with_flag(arr):

2 n ← length(arr)

3 swapped ← false

4 for i ← 0 to n - 1 do

5 swapped ← false
6 for j ← 0 to n - i - 1 do

7 if arr[j] > arr[j + 1] then

8 swap(arr[j], arr[j + 1])

9 swapped ← true
1
end
0
1
end
1
1
if swapped = false then
2
1
break
3
1
end
4
1
end
5
1
6 end
Selection Sort
Idea:
 The selection sort algorithm is a brute-force sorting algorithm
 The algorithm iterates through the array, finding the minimum
element in the unsorted part.
 It then swaps the minimum element with the element at the current
position.
Pseudo code:

1 Function SelectionSort(arr)
2 n ← length(arr)
3
4 for i ← 0 to n - 2 do
5 min ← i
6 for j ← i + 1 to n - 1 do
7 if arr[j] < arr[m] then
8 min ← j
9 end
1 end
0
1 if min != i then
1
1 swap(arr[i], arr[min])
2
1 end
3
1 end
4
1 end
5

Illustration:

Ste
Array Annotations
p
Find the minimum element, which is 1. Swap it with
{5, 3, 2,
0 the first element
1}
Find the minimum element from the remaining
{1, 3, 2,
1 unsorted part, which is 2. Swap it with the second
5}
element
Find the minimum element from the remaining
{1, 2, 3,
2 unsorted part, which is 3. Since it's already in
5}
the correct position, no swap is needed
Find the minimum element from the remaining
{1, 2, 3,
3 unsorted part, which is 5. Since it's already in
5}
the correct position, no swap is needed

Complexity:
Space:
The algorithm does not requires extra arrays, so space complexity is
always Θ(1) for all cases.
Time:

Algorithms Best-case Worst-case Stable

Selection Sort O(n2) O(n2) No

 Basic operation: key comparison arr[j] < arr[m] in line 7.


 The number of times the basic operation is executed depends only
on n → the running time is the same for every input of a specific
n.
 The sum represents the total number of executions of the
algorithm’s basic operation:
n −2 n−1 n−2
n (n−1)
∑¿ ∑ 1 }=∑ ( n−1−i−1+1 ) =
2
¿
i=0 j=i+1 i=0
 Order of growth: O(n2)
Insertion Sort
Idea:
 The insertion sort algorithm is a decrease and conquer sorting
algorithm.
 The algorithm iterates through the array, starting from the
second element.
 For each element, it finds the correct position to insert it into
the already sorted subarray on the left.
Pseudo code:

1 Function InsertionSort(arr)
2 n ← length(arr)
3 for i ← 1 to n - 1 do
4 key ← arr[i]
5 j ← i - 1
6 while j >= 0 and arr[j] > key do
7 arr[j + 1] ← arr[j]
8 j ← j - 1
9 end
1 arr[j + 1] ← key
0
1 end
1
1 end
2

Illustration:

Ste
Array Annotations
p
{5, |3, 2,
0
1}
Take the element (3), compare it with the sorted
{3, 5, |2,
1 portion (5) and swap them.
1}
Consider the next element (2). Compare it with the
{2, 3, 5, | sorted portion (3, 5) and swap it with (5). The
2
1} sorted portion is now {3, 2, 5}. Then swap it with
(3).
Find the minimum element from the remaining
unsorted part, which is 1. Compare it with the
3 {1, 2, 3, 5}
sorted portion (2,3, 5) and swap it with (5), (3)
then (2).
Complexity:
Space:
The algorithm does not requires extra arrays, so space complexity is
always Θ(1) for all cases.
Times:

Algorithms Best-case Worst-case Stable

Insertion Sort O(n) O(n2) No

 Basic operation: key comparison arr[j] > key in line 6.


 The number of times the basic operation is executed depends on n
and the nature of input
Worst-case: Descending sorted array
 Sum of number of times the basic operation for the worst-case is:
n −1 i −1 n−1
n (n−1)
∑ ¿ ∑ 1 }=∑ i= 2
¿
i=1 j=0 i=1
 Order of growth: O(n2)
Best-case: Ascending sorted array
 Sum of number of times the basic operation for the best-case is:
n −1

∑ 1=n−1
i=1
 Order of growth: O(n)
Optimize: Insertion sort with binary search
 Instead of sequentially comparing each element, insertion binary
sort uses binary search to find the appropriate insertion
position for each new element. This reduces the number of
comparisons from O(n) to O(log n).
 Similar to regular insertion sort, insertion binary sort
maintains a sorted subarray. Each time a new element is added, it
uses binary search to find the correct insertion position.
 After finding the appropriate insertion position using binary
search, insertion binary sort shifts the larger elements to the
right to make room for the new element.
Pseudo code:
1 Function binarySearch(arr, key, low, high)
2 if high <= low then
3 if key > arr[low] then
4 return low + 1
5 else
6 return low
7 end
8 end
9
1
mid = (low + high) / 2
0
1
1
1
if key == arr[mid] then
2
1
return mid + 1
3
1
end
4
1
5
1
if key > arr[mid] then
6
1
return binarySearch(arr, key, mid + 1, high)
7
1
else
8
1
return binarySearch(arr, key, low, mid - 1)
9
2
end
0
2
end
1
2
2
2
Function insertionSort(arr, n)
3
2
for i = 1 to n - 1 do
4
2
selected = arr[i]
5
2
loc = binarySearch(arr, selected, 0, i - 1)
6
2
7
2 for j = i - 1 down to loc do
8
2
arr[j + 1] = arr[j]
9
3
end
0
3
1
3
arr[loc] = selected
2
3
end
3
3
4 end
Shaker Sort (Cocktail Sort)
Idea:
 The main idea of Shaker Sort is to combine Bubble Sort in two
directions, from left to right and from right to left.
 The sorting process works as follows:
 Start from the first element, compare and swap if the next
element is larger.
 Then, start from the last element, compare and swap if the
previous element is larger.
 Repeat this process until the entire array is sorted.
 Shaker Sort is considered a variation of Bubble Sort, with the
advantage of reducing the number of comparisons and swaps
compared to the standard Bubble Sort.
Pseudo code:

1 Function ShakerSort(arr, n)
2 left = 0
3 right = n-1
4 temp = 0
5 sorted = true
6 while left < right do
7 for i from left to right-1 do
8 if arr[i] > arr[i+1] then
9 swap(arr[i], arr[i+1])
1 temp = i
0
1 sorted = false
1
1 end
2
1 end
3
1 if sorted = true then
4
1 break
5
1 end
6
1 sorted = true
7
1 right = right - 1
8
1 for i from right-1 to left do
9
2 if arr[i] < arr[i-1] then
0
2 swap(arr[i], arr[i-1])
1
2 temp = i
2
2 sorted = false
3
2 end
4
2 end
5
2 left = left + 1
6
2 if sorted = true then
7
2 break
8
2 end
9
3 end
0
3 end
1

Illustration:

Ste
Array Annotations
p

0 {5, 2, 3, 1}
1 {2, 5, 3, 1} Start the first pass, moving from left to right:
Compare (5, 2), swap => {2, 5, 3, 1}
2 {2, 3, 5, 1}
Compare (5, 3), swap => {2, 3, 5, 1}
3 {2, 3, 1, 5}
Compare (5, 1), swap => {2, 3, 1, 5}
4 {2, 1, 3, 5}
Compare (3, 1), swap => {2, 1, 3, 5}
5 {1, 2, 3, 5}
Compare (2, 1), swap => {1, 2, 3, 5}
6 {1,2,3,5} Start the second pass, moving from right to left:
Compare (2, 3), no swap

Complexity:
Space:
The algorithm does not requires extra arrays, so space complexity is
always Θ(1) for all cases.
Time:
Time:

Algorithms Best-case Worst-case Stable

Shaker Sort O(n) O(n2) Yes

 Base operator is comparison in line 8 and 20.


 The number of times the basic operation is executed depends on n
and the nature of input.
Worst-case
 Order of growth: O(n2)
Best-case
 In the best-case, the input array is already sorted in ascending
or descending order.
 We can use an additional flag like BubbleSort_with_flag that we
have already mentioned. This improvement will make the best-case
time complexity of Bubble Sort to be O(n).
Shell sort
Idea:
 Shell Sort is an improvement over the simple Insertion Sort
algorithm, with better efficiency for larger arrays
 The main idea of Shell Sort is to use the difference in the gap
between the elements being compared and swapped.
 The algorithm starts with a large gap (usually n/2, where n is
the size of the array) and gradually reduces the gap down to 1.
 In each iteration, the algorithm uses Insertion Sort on the
elements separated by the current gap.
 When the gap reaches 1, the algorithm has sorted the entire
array.
Pseudo code:

1 Function ShellSort(arr, n)
2 for gap = n/2 to 0 do
3 for i from gap to n-1 do
4 temp = arr[i]
5 j = i
6 while j >= gap and arr[j-gap] > temp do
7 arr[j] = arr[j-gap]
8 j = j - gap
9 end
1 arr[j] = temp
0
1 end
1
1 end
2
1 end
3

Illustration:

Step Array Annotations

0 {5, 2, 3, 1}

{|2, 5, 3, For the initial gap of 2, compare (5, 2), swap =>
1
1} {2, 5, 3, 1}
{2, 5, |1,
2
3} Compare (3, 1), swap => {2, 5, 1, 3}
{2, |5, 1, Repeat the sorting process with the new gap size of
3
3} 1, compare (2, 5), no swap => {2, 5, 1, 3}
{1, 2, |5,
4
3} Compare (5), (2) with (1), swap => {1, 2, 5, 3}
{1, 2, 3, | Compare (1), (2), (5) with (3), swap => {1, 2, 3,
5
5} 5}

Complexity:
Space:
The algorithm does not requires extra arrays, so space complexity is
always Θ(1) for all cases.
Times:

Algorithms Best-case Worst-case Stable

Shell Sort O(nlogn) O(n2) No

 Base operator is comparison in line 8 and 20.


 The number of times the basic operation is executed depends on n
and the nature of input.
Worst-case
 The worst-case occurs in a sorting algorithm when the elements to
be sorted are in reverse order
 Order of growth: O(n2)
Best-case
 In the best-case, the input array is already sorted in ascending
or descending order.
 Order of growth: O(nlogn).
Merge Sort
Idea:
 The merge sort algorithm is a divide-and-conquer sorting
algorithm.
 The main idea of Merge Sort is to divide the input array
recursively into smaller subarrays, sort them, and then merge
them back together.
 The SplitMS function is the recursive part of the algorithm,
where it divides the array into two halves and recursively sorts
each half.
 The Merge function is responsible for merging the two sorted
subarrays back together. It compares the elements from the two
subarrays and places them in the correct order in a temporary
array.
 After the two subarrays are merged, the elements in the temporary
array are copied back to the original array.
Pseudo code:

1 Function mergeSort(arr, n)
2 mergeSort_I(arr, 0, n - 1)
3 end
4
5 Function mergeSort_I(arr, l, r)
6 if l >= r then
7 return
8 mid ← l + (r - l) / 2
9 mergeSort_I(arr, l, mid)
1
mergeSort_I(arr, mid + 1, r)
0
1
merge(arr, l, mid, r)
1
1
end
2
1
3
1
Function merge(arr, l, mid, r)
4
1
left_size ← mid - l + 1
5
1
right_size ← r - mid
6
1
leftArr ← new array of size left_size
7
1
rightArr ← new array of size right_size
8
1
9
2
for i ← 0 to left_size - 1 do
0
2
leftArr[i] ← arr[l + i]
1
2
end
2
2
3
2
for j ← 0 to right_size - 1 do
4
2
rightArr[j] ← arr[mid + 1 + j]
5
2
end
6
2
7
2
left_idx ← 0, right_idx ← 0, merged_idx ← l
8
2
9
3
while left_idx < left_size and right_idx < right_size do
0
3
if leftArr[left_idx] <= rightArr[right_idx] then
1
3
arr[merged_idx] ← leftArr[left_idx]
2
3
left_idx ← left_idx + 1
3
3
else
4
3
arr[merged_idx] ← rightArr[right_idx]
5
3
right_idx ← right_idx + 1
6
3
end
7
3
merged_idx ← merged_idx + 1
8
3
end
9
4
0
4
while left_idx < left_size do
1
4
arr[merged_idx] ← leftArr[left_idx]
2
4
left_idx ← left_idx + 1
3
4
merged_idx ← merged_idx + 1
4
4
end
5
4
6
4
while right_idx < right_size do
7
4
arr[merged_idx] ← rightArr[right_idx]
8
4
right_idx ← right_idx + 1
9
5
merged_idx ← merged_idx + 1
0
5
end
1
5
2 end

Illustration:

Step Array Annotations

0 {5, 2, 4, 0, 6, 1}
Divide the array into two
1 {5, 2, 4}, {0, 6, 1}
halves
Divide the array into two
2 {5,2}, {4}, {0,6}, {1}
halves
Divide the array into two
3 {5}, {2}, {4}, {0}, {6}, {1}
halves
4 {2,5},{4}, {0,6},{1}
Merge two halves.
5 {2, 4,5}, {0,1,6}
Merge two halves.
6 {0, 1, 2, 4, 5, 6}
Merge two halves.

Complexity:

Algorithms Best-case Worst-case Stable

Merge Sort O(nlogn) O(nlogn) Yes


Space:
 Each recursion Merge step make an extra temp array to copy data,
space complexity is O(n) for all cases.
Time:
 Basic operations: 2SplitMs + 1Merge
 The number of times the basic operation is:

C ( n )=2 C ( n2 )+ Cmerge ( n) for n>1 ,C ( 1)=0


 The number of key comparisons in Merge depends on the nature of 2
subarrays. In the worst-case:
 When the number of comparisons is maximum that means every
element of array will be compared at least once (Storing alternate
elements in left and right subarray):
Cmerge ( n )=n−1
 According to Master Theorem, C(n) ∈ O(nlogn)
Heap Sort
Idea:
 The heap sort algorithm is a divide-and-conquer sorting
algorithm.
 The main idea of Heap Sort is to use the Max-Heap data structure
to sort the input array.
 The algorithm first builds a Max-Heap from the input array using
the MaxHeapify function.
 In the MaxHeapify function, the algorithm ensures that the root
of the subtree at index i is the largest element by comparing it
with its left and right children and swapping if necessary.
 After building the Max-Heap, the algorithm repeatedly extracts
the maximum element (the root of the heap) and places it at the
end of the sorted portion of the array.
 This process is repeated until the entire array is sorted.
Pseudo code:

1 Function HeapSort(arr)
2 n = length(arr)
3 // Build Max Heap
4 for i from floor(n/2) - 1 to 0
5 MaxHeapify(arr, i, n)
6 end
7
8 // Second Stage
9 for i from n-1 to 1
1 swap(arr[0], arr[i])
0
1 MaxHeapify(arr, 0, i)
1
1 end
2
1 end
3
1
4
1 Function MaxHeapify(arr, i, n)
5
1 left = 2*i + 1
6
1 right = 2*i + 2
7
1 largest = i
8
1 if left < n and arr[left] > arr[largest] then
9
2 largest = left
0
2 end
1
2 if right < n and arr[right] > arr[largest] then
2
2 largest = right
3
2 end
4
2 if largest != I then
5
2 swap(arr[i], arr[largest])
6
2 MaxHeapify(arr, largest, n)
7
2 end
8
2 end
9

Illustration:

Nod
Array Annotations
e
{5, 2, 4, 0, Build the Heap, starting with the index 3 (n/2 -
2
6, 1, 2} 1). 4 is at right position.
{5, 2, 4, 0,
1 2 is smaller than 6, so 2 will be swapped with 6.
6, 1, 2}
{5, 6, 4, 0,
1 2 does not have any leaf, no swap.
2, 1, 2}
{5, 6, 4, 0,
0 5 is smaller than 6, so 5 will be swapped with 6.
2, 1, 2}
{6, 5, 4, 0, 5 is at right position. Extract the maximum
0
2, 1, 2} element and rebuild the Heap.
{2, 5, 4, 0,
0 2 is smaller than 5 so 2 will be swapped with 5.
2, 1 | 6}
{5, 2, 4, 0, 2 is at right position. Extract the maximum
0
2, 1 | 6} element and rebuild the Heap.
{1, 2, 4, 0, 2
0 1 is smaller than 4 so 1 will be swapped with 4.
| 5, 6}
{4, 2, 1, 0, 2 1 does not have any leaf, no swap. Extract the
0
| 5, 6} maximum element and rebuild the Heap.
{2, 2, 1, 0 | 2 is at the right position so no swap. Extract the
0
4, 5, 6} maximum element and rebuild the Heap.
{0, 2, 1 | 2,
0 0 is smaller than 2 so 0 will be swapped with 2.
4, 5, 6}
{2, 0, 1 | 2, 0 does not have any leaf, no swap. Extract the
0
4, 5, 6} maximum element and rebuild the Heap.
{1, 0 | 2, 2, 1 is at the right position, no swap. Extract the
0
4, 5, 6} maximum element and rebuild the Heap.
(0 | 1, 2, 2,
0 Extract the maximum element and rebuild the Heap.
4, 5, 6}
{0, 1, 2, 2,
0
4, 5, 6}
Complexity:
Space:
The algorithm does not requires extra arrays, so space complexity is
always Θ(1) for all cases.
Time:

Algorithms Best-case Worst-case Stable

Heap Sort O(nlogn) O(nlogn) No

 The number of times the basic operation is executed depends on n


and the nature of input. But the complexity of worst-case and
best-case for heap sort are both O(nlogn).
 The heap construction stage(Max_heapify and build max heap):
O(nlogn)
 The second stage: the number of comparisons in n-1 times calling
Max-heapify is: (n-1)* O(nlogn) ∈ O(nlogn)
 For both stages, we get: O(nlogn) + O(nlogn) = O(nlogn).
Quick Sort
Idea:
 The quick sort algorithm is a divide-and-conquer sorting
algorithm.
 Select a pivot element, which is typically the middle element or
the first/last element of the current partition.
 Partition the array into two parts: one part with elements less
than the pivot, and the other part with elements greater than or
equal to the pivot.
 Recursively apply the quick sort algorithm to the two partitions.
 For small partitions (less than 10 elements), use insertion sort
instead of recursive quick sort to improve efficiency.
Pseudo code:

1 Function Sort_First_Middle_Last(arr, left, right)


2 mid = (left + right) / 2
3 if (arr[left] > arr[mid]) then
4 Swap(arr[left], arr[mid])
5 if (arr[mid] > arr[right]) then
6 Swap(arr[mid], arr[right])
7 if (arr[left] > arr[right]) then
8 Swap(arr[left], arr[right])
9 return mid
1
0
1 Function Partition(arr, left, right)
1
1 pivot = Sort_First_Middle_Last(arr, left, right)
2
1 Swap(arr[pivot], arr[right-1])
3
1 pivot = right - 1
4
1
5
1 left = left + 1
6
1 right = right - 2
7
1
8
1 i = left
9
2 while (left <= right) do
0
2 while arr[pivot] < arr[right] and left <= right do
1
2 right = right - 1
2
2 end
3
2 while arr[pivot] > arr[left] and left <= right do
4
2 left = left + 1
5
2 end
6
2 if (left <= right) then
7
2 Swap(arr[right], arr[left])
8
2 left = left + 1
9
3 right = right - 1
0
3 end
1
3 end
2
3 Swap(arr[pivot], arr[left])
3
3 return left
4
3
5
3 Function Qs_recursion(arr, left, right)
6
3 if (right - left + 1 >= 10) then
7
3 pivot = Partition(arr, left, right)
8
3 QS_Recursion(arr, left, pivot - 1)
9
4 QS_Recursion(arr, pivot + 1, right)
0
4 end
1
4 else then
2
4 for i from left+1 to right-left do
3
4 key = arr[i]
4
4 j = i - 1
5
4 while j >= left and arr[j] > key do
6
4 arr[j+1] = arr[j]
7
4 j = j - 1
8
4 end
9
5 arr[j+1] = key
0
5 end
1
5 end
2
5 end
3
5 Function QuickSort(arr, n)
4
5 QS_Recursion(arr, 0, n-1)
5
5 end
6

Illustration:

Ste
Array Annotations
p
{5, 2, 4, 0, 9, 3, Mid = 7 (1) (mid= (last (15) + first
0 7, 1, 8, 5, 10, 8, (0)) /2) is smaller than 2 and 2 is
1, 6, 1, 2} smaller than 5, 2 is pivot
{1, 1, 1, 0}, 2 {3,
1 7, 4, 8, 5, 10, 8,
2, 6, 5, 9} The array is partitioned.
{0,1,1,1}, 2, {3, The left sub-array is sorted by insertion
2 7, 4, 8, 5, 10, 8, sort. Mid = 10(10) (last = 15(9), first =
2, 6, 5, 9} 5(3)). Pivot is 9.
{0,1,1,1}, 2, {3,
5 is smaller than 6, so 5 will be swapped
3 7, 4, 8, 5, 10, 8,
with 6.
2, 6, 5, 9}
{0,1,1,1}, 2, {3,
5 is at right position. Extract the
4 7, 4, 8, 5, 5, 8,
maximum element and rebuild the Heap.
2, 6, 10, 9}
{0,1,1,1}, 2, {3,
The left sub-array is sorted by insertion
5 7, 4, 8, 5, 5, 8,
sort.
2, 6), 9, {10}
{0, 1, 1, 1, 2, 2,
6 3, 4, 5, 5, 6, 7,
8, 8, 9, 10}
Complexity:
Space:
The algorithm does not requires extra arrays, so space complexity is
always Θ(1) for all cases.
Time:

Algorithms Best-case Worst-case Stable

Quick Sort O(nlogn) O(n2) No

 Basic operation: 2 QS_Recursion + 1 Partition


 The number of key comparisons in partition function depends only
on n.
 The number of QS_Recursion calls depend on pivot.
 The sum of basic operation are calculated for 2 cases:
 In the best-case, all the pivot splits in the middle of
corresponding subarrays:

C ( n )=C ( n2 )+n−1 for n>1 , C ( 1)=0 ∈O ( nlogn ) by Master Theorem


 In the worst-case, 1 of 2 subarrays is empty:

n ( n−1 )
C ( n )=n−1+n−2+ n−3+ …+1= ∈O(n 2)
2
Optimize: Smart Pivot Selection
 Instead of choosing a random pivot or using the first/last
element, select the pivot using other strategies such as:
 Choosing the median of three elements (first, middle, last)
 Median of medians algorithms
 Choosing a good pivot can help balance the recursion tree and
limit stack overflow.
Counting Sort
Idea:
 The counting sort algorithm is a non-comparative sorting
algorithm.
 Find the maximum element in the input array.
 Create a frequency array to store the count of each unique
element in the input array.
 Modify the frequency array such that each element at each index
stores the sum of the previous counts.
 Build the output array by iterating the input array in reverse
order and using the frequency array to decide the position of
each element.
 Lastly, copy the sorted elements from the output array to the
original input array.
Pseudo code:

1 Function CountingSort(a, n)
2 length ← 0
3
4 temp ← new vector<int>(n, 0)
5
6 // Find the maximum element in the array
7 for i ← 1 to n-1 do
8 if a[i] > a[length] then
9 length ← i
1 end
0
1 end
1
1
2
1 // Create a frequency array to store count of individual
3 digits
1 frequency ← new vector<int>(a[length] + 1, 0)
4
1 for i ← 0 to n-1 do
5
1 frequency[a[i] % 10] ← frequency[a[i] % 10] + 1
6
1 end
7
1
8
1 // Compute the cumulative frequency
9
2 for i ← 1 to a[length] do
0
2 frequency[i] ← frequency[i] + frequency[i-1]
1
2 end
2
2
3
2 // Place the elements in sorted order
4
2 for i ← n-1 to 0 do
5
2 temp[frequency[a[i] % 10] - 1] ← a[i]
6
2 frequency[a[i]] ← frequency[a[i]] - 1
7
2 end
8
2
9
3 // Overwrite the original array
0
3 for i ← 0 to n-1 do
1
3 a[i] ← temp[i]
2
3 end
3
3 end
4

Complexity:
Space:
Counting sort requires array to store the frequency of numbers
(smallest element – largest element), so space complexity is O(k) (k
is the largest element)
Time:

Algorithms Best-case Worst-case Stable

Counting Sort O(n+k) O(n+k) Yes

 Basic operation: assignments inside 4 loops.


 The number of key comparisons depends on the array size and the
max value of the array.
 Sum of number of times the basic operations executed is:
C ( n , k )=k +1+ n+k +n+ n=2 k +3 n+ 1
 Order of growth: 𝑶(𝒏 + 𝒌)
Radix Sort:
Idea:
 The radix sort algorithm is a non-comparative sorting algorithm.
 Find the maximum element in the input array to know the number of
digits.
 Do counting sort for every digit. Instead of moving digits, exp
is changed. exp is 1 (one's place), 10 (ten's place), 100
(hundred's place), and so on.
 Repeat step 2 until the maximum digit in the array is reached.
Pseudo code:

1 Function RadixSort(arr, n)
2 max ← GetMax(arr, n)
3
4 for exp ← 1 to max/10 do
5 frequency ← new vector<int>(10, 0)
6 temp ← new vector<int>(n, 0)
7
8 for i ← 0 to n-1 do
frequency[(arr[i]/exp)%10] ← frequency[(arr[i]/exp)
9
%10] + 1
1
end
0
1
1
1
2 for i ← 1 to 9 do
1
frequency[i] ← frequency[i] + frequency[i-1]
3
1
end
4
1
5
1
for i ← n-1 to 0 do
6
1
temp[frequency[(arr[i]/exp)%10]-1] ← arr[i]
7
1 frequency[(arr[i]/exp)%10] ← frequency[(arr[i]/exp)
8 %10] - 1
1
end
9
2
0
2
for i ← 0 to n-1 do
1
2
arr[i] ← temp[i]
2
2
3 end
2
end
4
2
end
5
2
6
2
Function GetMax(arr, n)
7
2
max ← 0
8
2
9
3
for i ← 1 to n-1 do
0
3
1 if arr[i] > arr[max] then
3
max ← i
2
3
end
3
3
end
4
3
end
5

Illustration:

Ste
Array Annotations
p
{18, 29, 59,
0 The largest element has 2 digits
30, 99, 17}
{30, 17, 18, The array will be sorted based on their last
1 29, 39, 59, digit. The order of elements with have the same
99} last digit will be kept.
{17, 18, 29, The array will be sorted based on their second
2 10, 39, 59, last digit. The order of elements with have the
99} same last second digit will be kept.
{17, 18, 29,
3 10, 39, 59,
99}

Complexity:
Space:
Radix sort requires a extra array to store for the buckets used during
sorting and for sorted output. So the space complexity of Radix sort
is O(n+k)
Time:

Algorithms Best-case Worst-case Stable

Radix Sort O(n) O(n2) Yes

The time complexity of radix sort depends on the number of digits(d)


in the largest number.
In the worst-case, time complexity is O(d*(n+k)) is d is constant it
will be O(n). When all elements have the same digits, radix sort will
need to process each digit of each element.
In the best-case, time complexity is O(n) when all the elements of the
array only have one digit.
Flash Sort
Idea:
 Find the maximum (max) and minimum (min) values in the array.
 Divide the array into m buckets, where m = 0.45 * n (n is the
size of the array).
 Distribute the elements of the array into the buckets based on
the formula: k = floor((m-1) * (a[i] - min) / (max - min)), where
i is the index of the element.
 Sort the elements within each bucket using a different sorting
algorithm (here it's Insertion Sort).
 Merge the sorted buckets to create the final sorted array.
Pseudo code:

1 Function flashSort(arr, n)
2
3 min_arr ← arr[0], max_arr ← arr[0]
4 for i ← 1 to n - 1 do
5 if arr[i] < min_arr then
6 min_arr ← arr[i]
7 else if arr[i] > max_arr then
8 max_arr ← arr[i]
9 end
1
end
0
1
1
1
if max_arr == min_arr then
2
1
return
3
1
end
4
1
5
1
m ← 0.45 * n
6
1
7 if m <= 2 then
1
m←2
8
1
end
9
2
L ← array of m elements, all initialized to 0
0
2
for i ← 0 to n - 1 do
1
2 k ← (m - 1) * (arr[i] - min_arr) / (max_arr - min_arr)
2
2
L[k] ← L[k] + 1
3
2
4 end
2
for k ← 1 to m - 1 do
5
2
L[k] ← L[k] + L[k - 1]
6
2
end
7
2
8
2
i ← 0
9
3
count ← 0
0
3
k ← m - 1
1
3
while count < n do
2
3
while i >= L[k] do
3
3
4 i←i+1
3
5 k ← (m - 1) * (arr[i] - min_arr) / (max_arr - min_arr)
3
end
6
3
flash ← arr[i]
7
3
while i != L[k] do
8
3
k ← (m - 1) * (flash - min_arr) / (max_arr - min_arr)
9
4
swap(arr[L[k] - 1], flash)
0
4
L[k] ← L[k] - 1
1
4
count ← count + 1
2
4
end
3
4
end
4
4
5
4
6 for k ← 1 to m - 1 do
4
7 for i ← L[k - 1] + 1 to L[k] - 1 do
4
8 selected ← arr[i]
4
j ← i - 1
9
5
while j >= 0 and arr[j] > selected do
0
5
1 arr[j + 1] ← arr[j]
5
2 j←j-1
5
3 end
5
4 end
5
5 arr[j + 1] ← selected
5
6 end
5
7 return

Complexity:
Space:
Flash sort requires an extra array to store m buckets, which is 0.45 x
n. So that the space complexity of flash sort is O(m)
Time:

Algorithms Best-case Worst-case Stable

Flash Sort O(n) O(n2) No

 The best-case time complexity is O(n).


 However, the worst-case time complexity can be O(n2). This can
happen when all the elements to belong to one bucket. That leads
to the worst-case time complexity of insetion sort.

You might also like