排序算法--计数排序

唯一种没有比较的排序(指没有前后比较,还是有交换的)。统计每个元素出现的次数,直接计算元素在有序序列中的位置,要求数据是整数且范围有限。适用于 数据为小范围整数(如年龄、成绩),数据重复率较高时效率更优。可用于小范围整数排序、基数排序的底层排序( 作为基数排序的稳定排序子过程)、统计频率分布( 快速获取元素分布直方图)、海量数据预处理( 配合外部排序处理大数据文件**)**

以数组的下标当做数值,有这个数的时候a[i]++;

1绝对映射:

2相对映射:

cpp 复制代码
#include <stdlib.h>
#include <assert.h>

// 计数排序核心函数(稳定排序版本)
void countingSort(int arr[], int n) {
    if (n <= 1) return; // 无需排序

    // 1. 确定数据范围
    int max = arr[0], min = arr[0];
    for (int i = 1; i < n; i++) {
        if (arr[i] > max) max = arr[i];
        if (arr[i] < min) min = arr[i];
    }
    const int range = max - min + 1; // 实际数值范围

    // 2. 创建计数数组并初始化
    int* count = (int*)calloc(range, sizeof(int));
    assert(count != NULL);

    // 3. 统计每个元素出现次数
    for (int i = 0; i < n; i++) {
        count[arr[i] - min]++; // 偏移处理负数
    }

    // 4. 计算累计位置(保证稳定性)
    for (int i = 1; i < range; i++) {
        count[i] += count[i - 1];
    }

    // 5. 反向填充结果数组(关键稳定性操作)
    int* output = (int*)malloc(n * sizeof(int));
    assert(output != NULL);
    
    for (int i = n - 1; i >= 0; i--) {
        output[count[arr[i] - min] - 1] = arr[i];
        count[arr[i] - min]--;
    }

    // 6. 复制回原数组
    for (int i = 0; i < n; i++) {
        arr[i] = output[i];
    }

    // 7. 释放内存
    free(count);
    free(output);
}
cpp 复制代码
#include <stdio.h>
// 打印数组
void printArray(int arr[], int size) {
    for (int i = 0; i < size; i++) {
        printf("%d ", arr[i]);
    }
    printf("\n");
}

int main() {
    // 测试数据(包含负数)
    int arr[] = {-5, 2, -3, 4, 1, 2, 8, 5, 3, -1};
    int n = sizeof(arr) / sizeof(arr[0]);

    printf("排序前: ");
    printArray(arr, n);

    countingSort(arr, n);

    printf("排序后: ");
    printArray(arr, n);

    return 0;
}

优化建议:

1.通过min值偏移处理负数,支持全整数范围排序

2.通过反向遍历填充输出数组,保留相同元素的原始顺序,已保证稳定性

**3.**动态计算range值,避免不必要的内存浪费

cpp 复制代码
void countingSortSpaceOptimized(int arr[], int n) {
    // ...(省略范围计算步骤)...
    
    // 直接根据计数数组覆盖原数组(非稳定)
    int idx = 0;
    for (int i = 0; i < range; i++) {
        while (count[i]-- > 0) {
            arr[idx++] = i + min;
        }
    }
}
相关推荐
wfbcg5 分钟前
每日算法练习:LeetCode 76. 最小覆盖子串 ✅
算法·leetcode·职场和发展
Wect9 分钟前
LeetCode 149. 直线上最多的点数:题解深度剖析
前端·算法·typescript
qianpeng89712 分钟前
运动声源的到达结构仿真
算法
费曼学习法17 分钟前
线段树:区间查询的"终极武器",一文看透高效范围统计
算法
itman30117 分钟前
Windows环境下编译运行C语言程序的方法及工具选择
c语言·visualstudio·mingw·编译器·windows环境
wayz1124 分钟前
Day 2:线性回归原理与正则化
算法·机器学习·数据分析·回归·线性回归
QQ6765800835 分钟前
基于yolo26算法的水下目标检测图像数据集 海洋生物识别 海胆识别 海龟识别数据集 海洋生物监测与保护工作 潜水作业安全辅助系 水下环境感知第10408期
算法·目标检测·水下目标检测·海洋生物识别·海胆 海龟识别·海洋生物监测与保护工作·潜水作业安全辅助 水下环境感知
七颗糖很甜1 小时前
基于 OpenCV 的 FY2 云顶图云块追踪算法实现
人工智能·opencv·算法
__Wedream__1 小时前
NTIRE 2026 Challenge on Efficient Super-Resolution——冠军方案解读
人工智能·深度学习·算法·计算机视觉·超分辨率重建
yong15858553431 小时前
Linux C++ 中的 volatile变量在多线程环境下进行运算的问题
c语言·c++