Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Added Iterative Quick Sort #2684

Merged
merged 14 commits into from
Sep 4, 2024
Merged
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
132 changes: 132 additions & 0 deletions sorting/quick_sort_iterative.cpp
Original file line number Diff line number Diff line change
@@ -0,0 +1,132 @@
/**
* @file
* @brief Quick Sort without recursion. This method uses the stack instead.
* Both recursive and iterative implementations have O(n log n) best case
* and O(n^2) worst case.
* @details
* https://stackoverflow.com/questions/12553238/quicksort-iterative-or-recursive
* https://en.wikipedia.org/wiki/Quicksort
* https://www.geeksforgeeks.org/iterative-quick-sort/
* @author [Sebe324](https://github.com/sebe324)
*/

#include <iostream> /// for std::cout
#include <vector> /// for std::vector
#include <stack> /// for std::stack
#include <algorithm> /// for std::is_sorted
#include <cassert> /// for assert

sebe324 marked this conversation as resolved.
Show resolved Hide resolved

/**
* @namespace sorting
* @brief Sorting algorithms
*/
namespace sorting {
/**
* @brief The partition function sorts the array from
* start to end and uses the last element as the pivot.
* @param arr the array to be sorted
* @param start starting index
* @param end ending index
* @return int next index of the pivot
*/
int partition(std::vector<int> &arr, int start, int end)
{
int pivot = arr[end];
int index = start - 1;

for (int j = start; j < end; j++) {
if (arr[j] <= pivot) {
std::swap(arr[++index], arr[j]);
}
}

std::swap(arr[index + 1], arr[end]);
return index + 1;
}

/**
* @brief The main sorting function
* @details The iterative quick sort uses
* the stack instead of recursion for saving
* and restoring the environment between calls.
* It does not need the end and start params, because
* it is not recursive.
* @param arr array to be sorted
* @return void
*/
void iterativeQuickSort(std::vector<int> &arr)
{
std::stack<int> stack;
int start = 0;
int end = arr.size()-1;
stack.push(start);
stack.push(end);

while(!stack.empty())
{
end = stack.top();
stack.pop();
start = stack.top();
stack.pop();

int pivotIndex = partition(arr,start,end);

if(pivotIndex -1 > start)
{
stack.push(start);
stack.push(pivotIndex-1);
}

if(pivotIndex+1<end)
{
stack.push(pivotIndex+1);
stack.push(end);
}
}
}

sebe324 marked this conversation as resolved.
Show resolved Hide resolved
} // namespace sorting
/**
* @brief Self-test implementations
* @returns void
*/
void tests()
{
//TEST 1 - Positive numbers
std::vector<int> case1={100,534,1000000,553,10,61,2000,238,2756,9,12,56,30};
std::cout<<"TEST 1\n";
std::cout<<"Before: \n";
for(auto x : case1) std::cout<<x<<",";
std::cout<<"\n";
sorting::iterativeQuickSort(case1);
assert(std::is_sorted(std::begin(case1),std::end(case1)));
std::cout<<"Test 1 succesful!\n";
std::cout<<"After: \n";
for(auto x : case1) std::cout<<x<<",";
std::cout<<"\n";

//TEST 2 - Negative numbers
std::vector<int> case2={-10,-2,-5,-2,-3746,-785,-123, -452, -32456};
std::cout<<"TEST 2\n";
std::cout<<"Before: \n";
for(auto x : case2) std::cout<<x<<",";
std::cout<<"\n";
sorting::iterativeQuickSort(case2);
assert(std::is_sorted(std::begin(case2),std::end(case2)));
std::cout<<"Test 2 succesful!\n";
std::cout<<"After: \n";
for(auto x : case2) std::cout<<x<<",";
std::cout<<"\n";
}


/**
* @brief Main function
* @returns 0 on exit
*/
int main()
{
tests(); // run self test implementation
return 0;
}
Loading