C/C++基数排序

基数排序

基数排序是按照低位先排序,然后收集;再按照高位排序,然后再收集;依次类推,直到最高位。有时候有些属性是有优先级顺序的,先按低优先级排序,再按高优先级排序。最后的次序就是高优先级高的在前,高优先级相同的低优先级高的在前。

算法描述

  • 取得数组中的最大数,并取得位数;
  • arr为原始数组,从最低位开始取每个位组成radix数组;
  • 对radix进行计数排序(利用计数排序适用于小范围数的特点);

动态演示

请添加图片描述

代码实现

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

using namespace std;

int maxbit(vector<int>& data, int n) //辅助函数,求数据的最大位数
{
    int maxData = data[0];           // 最大数
    /// 先求出最大数,再求其位数,这样有原先依次每个数判断其位数,稍微优化点。
    for (int i = 1; i < n; ++i)
    {
        if (maxData < data[i])
            maxData = data[i];
    }
    int d = 1;
    int p = 10;
    while (maxData >= p)
    {
        //p *= 10; // Maybe overflow
        maxData /= 10;
        ++d;
    }
    return d;
}
void radixsort(vector<int>& data) //基数排序
{
    int n = data.size();
    int d = maxbit(data, n);
    int* tmp = new int[n];
    int* count = new int[10]; //计数器
    int i, j, k;
    int radix = 1;
    for (i = 1; i <= d; i++) //进行d次排序
    {
        //每次分配前清空计数器
        for (j = 0; j < 10; j++)
            count[j] = 0; 
        //统计每个桶中的记录数
        for (j = 0; j < n; j++)
        {
            k = (data[j] / radix) % 10; 
            count[k]++;
        }
        //将tmp中的位置依次分配给每个桶,求前缀和
        for (j = 1; j < 10; j++)
            count[j] += count[j - 1]; 
        //将所有桶中记录依次收集到tmp中
        for (j = n - 1; j >= 0; j--) 
        {
            k = (data[j] / radix) % 10;
            tmp[count[k] - 1] = data[j];
            count[k]--;
        }
        //将临时数组的内容复制到data中
        for (j = 0; j < n; j++) 
            data[j] = tmp[j];
        //位数加1
        radix = radix * 10;
    }
    delete[]tmp;
    delete[]count;
}

void print_arr(vector<int>& arr) {
	int n = arr.size();
	for (int i = 0; i < n; i++) {
		cout << arr[i] << "  ";
	}
	cout << endl;
}
int main() {
	vector<int> arr = { 3,44,38,5,47,15,36,26,27,2,46,4,19,50,48};
	cout << "排序前:" << endl;
	print_arr(arr);
    radixsort(arr);
	cout << "排序后:" << endl;
	print_arr(arr);
	system("pause");
	return 0;
}

结果输出

在这里插入图片描述

  • 2
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值