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

fix radix sort #454

Open
wants to merge 1 commit into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
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
59 changes: 57 additions & 2 deletions cpp/include/algorithm/sorting/radix_sort.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -85,12 +85,17 @@ int max_in_vector(const vector<int>& values) {
*/
void radix_sort_internal(vector<int>& values, const int mult_factor, const int add_factor,
const bool to_show_state = false) {





int max_value = max_in_vector(values);

// On each iteration of the following loop, extractor helps in getting the
// next significant digit, which is (value / extractor) mod 10
for (int extractor = 1; max_value / extractor > 0; extractor *= 10) {
count_sort(values, extractor, to_show_state, mult_factor, add_factor);
count_sort(values, extractor, mult_factor, add_factor, to_show_state);
}
}

Expand All @@ -104,7 +109,57 @@ void radix_sort(vector<int>& values, const int order = 1, const bool to_show_sta
add_factor = 9;
}

radix_sort_internal(values, mult_factor, add_factor, to_show_state);
// core radix sort only works on positive values
auto negs = std::any_of(values.begin(), values.end(), [](int v) {return v < 0; });

if (negs) {
// we have negative values so
// - split into neg and pos
// - sort each half
// - join together
auto negcount = std::count_if(values.begin(), values.end(), [](int v) {return v < 0; });

vector<int> pos;
vector<int> neg;

pos.reserve(values.size() - negcount);
neg.reserve(negcount);

// split
for (auto v : values) {
if (v < 0)
neg.push_back(-v);
else
pos.push_back(v);
}

// sort each half
radix_sort(neg, -order, to_show_state);
radix_sort(pos, order, to_show_state);

values.clear();

// join
if (order == 1) {
for (auto v : neg) {
values.push_back(-v);
}
for (auto v : pos) {
values.push_back(v);
}
}
else {
for (auto v : pos) {
values.push_back(v);
}
for (auto v : neg) {
values.push_back(-v);
}
}
}
else {
radix_sort_internal(values, mult_factor, add_factor, to_show_state);
}
}

#endif // RADIX_SORT_HPP
4 changes: 2 additions & 2 deletions cpp/test/algorithm/sorting/sorting.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -34,7 +34,7 @@ TEST_CASE("Sort in ascending order", "[sorting]") {
insertion_sort,
merge_sort,
quick_sort,
// radix_sort, // This test reveals that the radix sort is broken
radix_sort, // This test reveals that the radix sort is broken
selection_sort,
shell_sort
};
Expand Down Expand Up @@ -66,7 +66,7 @@ TEST_CASE("Sort in descending order", "[sorting]") {
insertion_sort,
merge_sort,
quick_sort,
// radix_sort, // This test reveals that the radix sort is broken
radix_sort, // This test reveals that the radix sort is broken
selection_sort,
shell_sort
};
Expand Down
Loading