Lesson 7
Advanced Applications of Sorting Algorithms in C++
Introduction to the Lesson

Welcome to this insightful session, where our aim is to master the complexities of the illustrious applications of sorting algorithms. Today's voyage links two problems: "Find the K-th smallest element in a List" and "Count the Number of Inversions in a List". These problems mirror practical scenarios, and the efficient techniques used to solve them provide valuable demonstrations of the application of sorting algorithms. By solving these two problems, we'll see how Quick Sort and Merge Sort knowledge applies here and helps provide efficient implementations for both questions.

Let's dive into these captivating problems!

Problem 1: Find the K-th Smallest Element in a List

Our first problem presents a list of integers and the number k. The challenge is finding the k-th smallest element in that given list. To further elucidate, k starts from 1, so for k = 1, you are seeking to find the smallest element; if k = 2, you're searching for the second smallest element, and so on. By the conclusion of this lesson, you'll be highly skilled at performing this task!

Problem 1: Naive Approaches

A primary instinctive solution could involve iteratively identifying and discarding the smallest element from the list until you reach the k-th smallest element. While it sounds straightforward, this approach, unfortunately, incurs high time complexity due to the repetitive scans of the list to locate the smallest element. This solution has an O(n2)O(n^2) complexity.

Another straightforward solution is just to sort the input array and return the k-th element:

C++
1#include <algorithm> 2#include <vector> 3 4int findKthSmallestNaive(std::vector<int>& inputArray, int k) { 5 std::sort(inputArray.begin(), inputArray.end()); 6 return inputArray[k - 1]; 7}

This approach has O(nlogn)O(n \log n) complexity and is as simple as two lines of code. However, there are more efficient approaches than this, as there is an O(n)O(n) solution to this problem, using Quick Sort techniques we covered earlier in this course.

Problem 1: Efficient Approach Explanation

Sorting steps in here to offer an efficient solution! The Quick Sort algorithm, a splendid application of divide and conquer, can solve this problem more efficiently. By selecting the right pivot for partitioning, the input list is divided into two parts: a left partition, which contains elements less than the pivot, and a right partition, which includes elements greater than the pivot.

If the pivot's position after elements are repositioned is the same as k, we have the k-th smallest element. If k is less than the pivot's position, the task is carried forward on the left partition; otherwise, on the right partition.

Problem 1: Solution Building – Partition

Our C++ solution mirrors this efficient approach as follows. Firstly, we define the partition procedure, the same way we do it in the quicksort:

C++
1#include <vector> 2#include <utility> // For std::swap 3 4int partition(std::vector<int>& arr, int start, int end) { 5 int pivot = arr[start]; 6 int i = start; 7 8 for (int j = start + 1; j <= end; j++) { 9 if (arr[j] <= pivot) { 10 i++; 11 std::swap(arr[i], arr[j]); 12 } 13 } 14 std::swap(arr[i], arr[start]); 15 return i; 16}
Problem 1: Solution Building – Main Logic

Then, we define the findKthSmallest function, essentially working as quicksort, but chasing a different goal:

C++
1#include <iostream> 2#include <vector> 3#include <climits> // For INT_MIN and INT_MAX 4 5int kthSmallest(std::vector<int>& arr, int start, int end, int k); 6 7int findKthSmallest(std::vector<int>& numbers, int k) { 8 if (numbers.empty() || numbers.size() < k) 9 return INT_MIN; 10 return kthSmallest(numbers, 0, numbers.size() - 1, k); 11} 12 13int kthSmallest(std::vector<int>& arr, int start, int end, int k) { 14 if (k > 0 && k <= end - start + 1) { 15 int pos = partition(arr, start, end); 16 17 if (pos - start == k - 1) { 18 return arr[pos]; 19 } 20 21 if (pos - start > k - 1) { 22 return kthSmallest(arr, start, pos - 1, k); 23 } 24 25 return kthSmallest(arr, pos + 1, end, k - pos + start - 1); 26 } 27 28 return INT_MAX; 29} 30 31int main() { 32 std::vector<int> numbers = {1, 7, 2, 4, 2, 1, 6}; 33 std::cout << findKthSmallest(numbers, 5) << std::endl; // It should print 4 34 return 0; 35}

After we form the partitions, we compare the pivot's position to k. If the positions match, our pivot is the k-th smallest element. If our k is smaller, we look at the left partition; otherwise, the right one.

Problem 2: Count the Number of Inversions in a List

Our second problem entails a list of integers; your task is to deduce the number of inversions in the list.

An inversion is a pair of elements where the larger element appears before the smaller one. In other words, if we have two indices i and j, where i < j and the element at position i is greater than the element at position j (numbers[i] > numbers[j]), we have an inversion.

For example, for numbers = {4, 2, 1, 3}, there are four inversions: (4, 2), (4, 1), (4, 3), and (2, 1).

Problem 2: Efficient Approach Explanation

In our quest for efficiency, the Merge Sort algorithm comes into play. At its core, Merge Sort is a divide-and-conquer-based sorting algorithm, providing an optimal efficiency of O(nlogn)O(n \log n). However, we can cleverly modify this algorithm to count the number of inversions in the array while sorting it. This additional functionality doesn't impact the time complexity; therefore, it remains O(nlogn)O(n \log n). So, how does this work?

The process starts by dividing the array into two halves, similar to how Merge Sort operates. Then, we recursively sort both array halves and merge them back. Here comes the twist in the tale: while merging these sorted halves, we add additional counting logic to keep track of inversions.

As the halves are already sorted, if an element of the right half is smaller than that of the left half, it's inverted. This is because the element from the right half should have been after 'all' the remaining elements of the left half in a sorted array. Thus, we don't just have a single inversion here, we have as many inversions as there are remaining elements in the left half.

By counting these inversions at each merge and adding them up, we get the total number of inversions in the array.

Problem 2: Solution Building – Supporting Struct

First, let's define a supportive struct to store our counter for each subarray:

C++
1#include <vector> 2 3struct Result { 4 std::vector<int> sorted; 5 long inversions; 6 7 Result(std::vector<int> sorted, long inversions) 8 : sorted(sorted), inversions(inversions) {} 9};
Problem 2: Solution Building – Main Logic

Then, implement the merge sort that keeps track of the number of inversions:

C++
1#include <vector> 2#include <iostream> 3#include <algorithm> 4 5Result mergeAndCountInversions(const std::vector<int>& left, const std::vector<int>& right); 6 7Result countInversions(std::vector<int> arr) { 8 if (arr.size() <= 1) { 9 return Result(arr, 0); 10 } 11 12 size_t middle = arr.size() / 2; 13 std::vector<int> left(arr.begin(), arr.begin() + middle); 14 std::vector<int> right(arr.begin() + middle, arr.end()); 15 16 Result leftResult = countInversions(left); 17 Result rightResult = countInversions(right); 18 Result result = mergeAndCountInversions(leftResult.sorted, rightResult.sorted); 19 20 return Result(result.sorted, leftResult.inversions + rightResult.inversions + result.inversions); 21} 22 23Result mergeAndCountInversions(const std::vector<int>& left, const std::vector<int>& right) { 24 std::vector<int> merged(left.size() + right.size()); 25 size_t i = 0, j = 0; 26 long inversions = 0; 27 28 for (size_t k = 0; k < merged.size(); ++k) { 29 if (i < left.size() && (j >= right.size() || left[i] <= right[j])) { 30 merged[k] = left[i++]; 31 } else { 32 merged[k] = right[j++]; 33 inversions += left.size() - i; 34 } 35 } 36 return Result(merged, inversions); 37} 38 39int main() { 40 std::vector<int> numbers = {4, 2, 1, 3}; 41 Result res = countInversions(numbers); 42 std::cout << "Number of inversions: " << res.inversions << std::endl; // It should print 4 43 return 0; 44}

While performing the merge in Merge Sort, we have added an additional counter to track inversions. If we come across an element in the right array that is smaller than an element in the left, we increment the counter by the number of items remaining in the left array. These all form inversions as per our problem's definition.

Lesson Summary

During today's lesson, we thoroughly inspected the advanced applications of Quick Sort and Merge Sort algorithms through the dissection of two exciting problems. We went from recognizing the issues, proposing naïve methods, progressing towards efficient approaches, and executing the C++ solutions.

Practice Exercises

You're now ready to flex those coding muscles! We have covered a substantial amount of theory and strongly advocate that real-world application solidifies the lessons learned. Be prepared for the forthcoming exercises, where you'll apply the logic imbibed in this session to similar problems. Let's get hands-on!

Enjoy this lesson? Now it's time to practice with Cosmo!
Practice is how you turn knowledge into actual skills.