该篇文章 所涉及代码收录仓库:登录 - Gitee.com
思想:计数排序又称为鸽巢原理,是对哈希直接定址法的变形应用。
操作步骤: 1. 统计相同元素出现次数 2. 根据统计的结果将序列回收到原来的序列中
a
一次,找到其中的最大值 max
和最小值 min
。这一步是为了确定数组中的数值范围,从而决定后续计数数组 count
的大小。range = max - min + 1
,并用 calloc
动态分配一个大小为 range
的整型数组 count
。计数数组的每个元素初始化为0,用于记录原数组中每个数值出现的次数。a
,对于数组中的每个元素 a[i]
,计算它与最小值的差值 a[i] - min
,并将计数数组中对应索引的位置加1。这样做是因为我们希望 count[0]
存储的是原数组中小于等于 min
的元素数量,count[1]
存储的是原数组中等于 min+1
的元素数量,依此类推,从而避免了因为负数或零而导致的索引错误。count
,对于 count[j]
非零的每个元素,将其对应的数值(即 j + min
)放回原数组 a
中,同时减少 count[j]
的计数。这里的循环保证了数值会按照原来的顺序被放置回数组,从而维持了稳定性排序。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\n");
return;
}
// 统计次数
for (int i = 0; i < n; i++)
{
count[a[i] - min]++;//min作为偏移量,然后再在对应数组位置++,统计该数字出现的次数
}
// 排序
int i = 0;
for (int j = 0; j < range; j++)
{
while (count[j]--)
{
a[i++] = j + min;
}
}
}
综上,计数排序在特定场景下(如数据范围不大、整数类型)是一种快速且高效的排序选择,但其适用场景相对有限,且空间效率较低。