1.计数排序思想
计数排序,顾名思义就是计算数据的个数
计数排序又称非比较排序
思想:计数排序又称为鸽巢原理,是对哈希直接定址法的变形应用。 操作步骤:
- 统计相同元素出现次数
- 根据统计的结果将序列回收到原来的序列中
计数排序的特性总结:
- 计数排序在数据范围集中时,效率很高,但是适用范围及场景有限。
- 时间复杂度:O(MAX(N,范围))
- 空间复杂度:O(范围)
2.计数排序过程
首先统计每个数据出现了多少次
假设有这么一个数组,下面的数组就是统计数据个数的
如果1出现,则对1位置++,如果3出现,则对3位置++,即
这里的代码核心稍微比较抽象,是在统计a数组中数据个数
接下来的操作是这样,对比count数组,0出现了0次,那就是0个0,1出现了3次,那就是3个1,其余同理,图示如下:
对比下来效率是非常高的,遍历一遍数组
同样,他也有局限性:
- 不适合分散的数据,更适合集中的数据
- 不适合浮点数、字符串、结构体数据排序,只适合整数
3.实现代码
先求最大值max和最小值min,然后遍历原数组统计次数,最后排序
#define _CRT_SECURE_NO_WARNINGS 1
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
void CountSort(int* a, int n)
{
int min = a[0], max = a[0];
for (int i = 1; i < n; i++)
{
if (a[i] < min)
min = a[i];
if (a[i] > max)
max = a[i];
}
int range = max - min + 1;
int* count = (int*)calloc(range, sizeof(int));
if (count == NULL)
{
printf("calloc fail!");
return;
}
for (int i = 0; i < n; i++)
{
count[a[i] - min]++;
}
int i = 0;
for (int j = 0; j < range; j++)
{
while (count[j]--)
{
a[i++] = j + min;
}
}
}
int main()
{
int a[] = { 10,11,10,11,15,1,2,3,5,4,2,1,0 };
int n = sizeof(a) / sizeof(a[0]);
for (int i = 0; i < n; i++)
{
printf("%d ", a[i]);
}
printf("\n");
CountSort(a, n);
for (int i = 0; i < n; i++)
{
printf("%d ", a[i]);
}
return 0;
}