Some short notes of CSW 2 (Sorting and Searching)
Sorting Short Notes
Below is a summary of general sorting techniques, including their working, best case, worst case, and best use cases:
Bubble Sort (Source 0):
- Working: Repeatedly steps through the list, compares adjacent elements, and swaps them if they are in the wrong order.
- Best case: O(n) when the array is already sorted.
- Worst case: O(n^2) when the array is reverse sorted.
- Best use: Small datasets or when the array is almost sorted.
Insertion Sort (Source 0):
- Working: Builds the final sorted array one item at a time by comparing and inserting the current element at its correct position in the sorted portion of the array.
- Best case: O(n) when the array is already sorted.
- Worst case: O(n^2) when the array is reverse sorted.
- Best use: Small datasets or when the array is almost sorted; also suitable for online sorting.
Selection Sort (Source 0):
- Working: Selects the smallest element in the remaining unsorted portion of the array and swaps it with the element in the current position.
- Best, average, and worst case: O(n^2) independent of data distribution.
- Best use: Small datasets, when memory usage is a concern.
Merge Sort (Source 0):
- Working: Divides the array into two halves, recursively sorts each half, and merges the sorted halves.
- Best, average, and worst case: O(n log n) independent of data distribution.
- Best use: Larger datasets, when a stable sort is required; however, it uses extra memory.
Quick Sort (Source 0):
- Working: Selects a 'pivot' element and partitions the array around the pivot, then recursively sorts the subarrays.
- Best and average case: O(n log n).
- Worst case: O(n^2) when the array is sorted or reverse sorted.
- Best use: Larger datasets, when a fast average-case performance is desired; however, it has poor worst-case performance.
Heap Sort (Source 0):
- Working: Builds a binary heap (max-heap) and repeatedly extracts the maximum element from the heap, then inserts it at the end of the sorted portion of the array.
- Best, average, and worst case: O(n log n) independent of data distribution.
- Best use: Larger datasets, when memory usage is a concern.
Radix Sort (Source 0):
- Working: Sorts integers based on their individual digits, starting from the least significant digit to the most significant digit.
- Best, average, and worst case: O(nk) where k is the maximum number of digits in elements of the array.
- Best use: When the range of input values is limited, and integers need to be sorted.
Count Sort (Source 0):
- Working: Counts the occurrences of each element in the array, then calculates the position of each element in the sorted array.
- Best, average, and worst case: O(n+k) where k is the size of the count array.
- Best use: When the range of input values is limited, and integers need to be sorted.
Bucket Sort (Source 0):
- Working: Divides the array into a finite number of buckets, then sorts each bucket individually using another sorting algorithm.
- Best and average case: O(n+k) where k is the number of buckets.
- Worst case: O(n^2) if all elements belong to the same bucket.
- Best use: When the input is uniformly distributed across a range, and a linear-time sorting algorithm is desired.
Java implementation of above Sorting algo
Here are Java implementations for each of the sorting techniques mentioned above:
- Bubble Sort (Source 0):
public static void bubbleSort(int[] arr) {
int n = arr.length;
for (int i = 0; i < n - 1; i++) {
for (int j = 0; j < n - 1 - i; j++) {
if (arr[j] > arr[j + 1]) {
int temp = arr[j];
arr[j] = arr[j + 1];
arr[j + 1] = temp;
}
}
}
}
- Insertion Sort (Source 0):
public static void insertionSort(int[] arr) {
int n = arr.length;
for (int i = 1; i < n; i++) {
int key = arr[i];
int j = i - 1;
while (j >= 0 && arr[j] > key) {
arr[j + 1] = arr[j];
j = j - 1;
}
arr[j + 1] = key;
}
}
- Selection Sort (Source 0):
public static void selectionSort(int[] arr) {
int n = arr.length;
for (int i = 0; i < n - 1; i++) {
int minIndex = i;
for (int j = i + 1; j < n; j++) {
if (arr[j] < arr[minIndex]) {
minIndex = j;
}
}
int temp = arr[minIndex];
arr[minIndex] = arr[i];
arr[i] = temp;
}
}
- Merge Sort (Source 0):
public static void mergeSort(int[] arr, int left, int right) {
if (left < right) {
int mid = (left + right) / 2;
mergeSort(arr, left, mid);
mergeSort(arr, mid + 1, right);
merge(arr, left, mid, right);
}
}
public static void merge(int[] arr, int left, int mid, int right) {
int n1 = mid - left + 1;
int n2 = right - mid;
int[] leftArr = new int[n1];
int[] rightArr = new int[n2];
for (int i = 0; i < n1; i++) {
leftArr[i] = arr[left + i];
}
for (int j = 0; j < n2; j++) {
rightArr[j] = arr[mid + 1 + j];
}
int i = 0, j = 0, k = left;
while (i < n1 && j < n2) {
if (leftArr[i] <= rightArr[j]) {
arr[k] = leftArr[i];
i++;
} else {
arr[k] = rightArr[j];
j++;
}
k++;
}
while (i < n1) {
arr[k] = leftArr[i];
i++;
k++;
}
while (j < n2) {
arr[k] = rightArr[j];
j++;
k++;
}
}
- Quick Sort (gfg)):
public static void quickSort(int[] arr, int low, int high) {
if (low < high) {
int pi = partition(arr, low, high);
quickSort(arr, low, pi - 1);
quickSort(arr, pi + 1, high);
}
}
public static int partition(int[] arr, int low, int high) {
int pivot = arr[high];
int i = low - 1;
for (int j = low; j < high; j++) {
if (arr[j] <= pivot) {
i++;
int temp = arr[i];
arr[i] = arr[j];
arr[j] = temp;
}
}
int temp = arr[i + 1];
arr[i + 1] = arr[high];
arr[high] = temp;
return i + 1;
}
Searching Short Notes
Here are some general searching techniques along with their working, best case, worst case, and best use cases:
Linear Search
- Working: Sequentially checks each element of the list until a match is found or the whole list has been searched.
- Best case: O(1) - item is found on the first comparison.
- Worst case: O(n) - item is not in the list or found at the last position.
- Best use: Suitable for small, unsorted datasets.
Binary Search
- Working: Requires a sorted dataset. It repeatedly divides the list in half and compares the middle element with the search value. If equal, the search is successful; otherwise, it continues in either the left or right half.
- Best case: O(1) - item is found at the middle of the list.
- Worst case: O(log n) - item is not in the list or found at one of the ends.
- Best use: Suitable for large, sorted datasets.
Java implementation of above Searching algo
Here are the Java implementations for each of the searching techniques mentioned above:
- Linear Search:
public class LinearSearch {
public static void main(String[] args) {
int[] nums = {2, 12, 15, 11, 7, 19, 45};
int target = 7;
System.out.println(search(nums, target));
}
static int search(int[] nums, int target) {
for (int index = 0; index < nums.length; index++) {
if (nums[index] == target) {
return index;
}
}
return -1;
}
}
- Binary Search:
public class BinarySearch {
public static void main(String[] args) {
int[] nums = {2, 12, 15, 17, 27, 29, 45};
int target = 17;
System.out.println(search(nums, target));
}
static int search(int[] nums, int target) {
int start = 0;
int end = nums.length - 1;
while (start <= end) {
int mid = start + (end - start) / 2;
if (nums[mid] > target)
end = mid - 1;
else if (nums[mid] < target)
start = mid + 1;
else
return mid;
}
return -1;
}
}