【C++算法】桶排序

桶排序(Bucket Sort)是一种分布式排序算法,它将待排序的数据分到有限数量的桶子里。每个桶子再个别排序(有可能再使用别的排序算法或是以递归方式继续使用桶排序进行排序)。桶排序是鸽巢排序(Pigeonhole Sort)的一种归纳结果。当要被排序的数组内的数值是均匀分布的时候,桶排序使用线性时间(Θ(n))。但桶排序并不是比较排序,它不受到 O(n log n) 下限的影响。

比如说我想要编写一个C++代码,输入10个随机数(0~9之间),现在我要给它们排序并输出

咋办呢?

首先,我们先建立一个数组num,长度为10(注意先memset为0)

然后,编写循环10次,让用户输入变量n。 

用户每次输入,num[n]都会+1,表示n的出现次数。

例如输入的十个数是:9977553311

首先,一开始收到的是9,num[9]++;

下一个又是9,第九项再加一

 

下一个是7,第七项加一

 

下一个还是7,第七项变成2

 

以此类推,最后结果应该是:

 

然后,该输出了。

 

我们从左向右遍历,0没有,不输出,1的值为2,循环输出2次

 

2没有,3值为2,循环输出2次。

 

4没有,5值为2,循环输出2次

 

6没有,7为2,循环输出2次

 

8没有,9值为2,循环输出两次

输入9977553311,输出应该如下图所示

 

总结

  1. 设置桶的数量:根据数据的分布和范围来设定桶的数量。
  2. 将数据放入桶中:遍历待排序的数组,将数据放入对应的桶中。
  3. 对每个桶内的数据进行排序:可以使用任何排序算法(如快速排序、插入排序等)对每个桶内的数据进行排序。
  4. 合并桶中的数据:按照桶的顺序,将桶内的数据依次取出,合并成一个有序数组。

示例代码

 
#include <iostream>
#include <vector>
#include <algorithm>

using namespace std;

// 假设数据都在 [0, MAX_NUM] 范围内
const int MAX_NUM = 100;

// 桶排序函数
void bucketSort(vector<int>& nums) {
if (nums.empty()) return;

// 1. 设置桶的数量
int bucketSize = max(1, (int)ceil((float)(MAX_NUM - nums.front()) / (nums.back() - nums.front()) * nums.size()));
vector<vector<int>> buckets(bucketSize);

// 2. 将数据放入桶中
for (int num : nums) {
int index = (num - nums.front()) * bucketSize / (nums.back() - nums.front());
buckets[index].push_back(num);
}

// 3. 对每个桶内的数据进行排序
for (auto& bucket : buckets) {
sort(bucket.begin(), bucket.end());
}

// 4. 合并桶中的数据
nums.clear();
for (auto& bucket : buckets) {
for (int num : bucket) {
nums.push_back(num);
}
}
}

int main() {
vector<int> nums = {4, 2, 8, 6, 1, 9, 5, 7, 3};
bucketSort(nums);

for (int num : nums) {
cout << num << " ";
}
cout << endl;

return 0;
}

注意事项

  • 桶排序对于数据分布均匀的情况效率较高,但如果数据分布不均匀,或者数据范围很大但数据量很小,桶排序可能效率不高。
  • 桶排序不是稳定的排序算法,因为合并桶时可能会改变相同元素的相对顺序。
  • 在实际应用中,可能需要根据数据的具体情况调整桶的数量和大小。
  • 在上面的示例代码中,我们假设了数据的范围在 [0, MAX_NUM] 之间,并使用了线性插值的方法来计算每个数据应该放入的桶的索引。但在实际应用中,你可能需要根据数据的实际情况来调整这个计算方法。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值