排序算法之计数排序

目录


一、简介

算法 平均时间复杂度 最好时间复杂度 最坏时间复杂度 空间复杂度 排序方式 稳定性
计数排序 O(n+k ) O(n+k) O(n+k) O(k) Out-place 稳定

稳定:如果A原本在B前面,而A=B,排序之后A仍然在B的前面;

不稳定:如果A原本在B的前面,而A=B,排序之后A可能会出现在B的后面;

时间复杂度: 描述一个算法执行所耗费的时间;

空间复杂度:描述一个算法执行所需内存的大小;

n:数据规模;

k:"桶"的个数;

In-place:占用常数内存,不占用额外内存;

Out-place:占用额外内存。

计数排序,又叫非比较排序。顾名思义,该算法不是通过比较数据的大小来进行排序的,而是通过统计数组中相同元素出现的次数,然后通过统计的结果将序列回收到原来的序列中。

核心在于将输入的数据值转化为键存储在额外开辟的数组空间中。作为一种线性时间复杂度的排序,计数排序要求输入的数据必须是有确定范围的整数。

算法步驟:

(1)找出待排序的数组中最大和最小的元素

(2)统计数组中每个值为i的元素出现的次数,存入数组C的第 i 项

(3)对所有的计数累加(从C中的第一个元素开始,每一项和前一项相加)

(4)反向填充目标数组:将每个元素i放在新数组的第C( i )项,每放一个元素就将C( i )减去1


二、代码实现

java 复制代码
public class CountingSort {
    public static void countingSort(int[] arr) {
        int len = arr.length;
        if (len < 2) return;
        int minVal = arr[0], maxVal = arr[0];
        for (int i = 1; i < len; i++) {
            if (arr[i] < minVal) {
                minVal = arr[i];
            } else if (arr[i] > maxVal) {
                maxVal = arr[i];
            }
        }

        int[] countArr = new int[maxVal - minVal + 1];
        for (int val : arr) {
            countArr[val - minVal]++;
        }
        for (int arrIdx = 0, countIdx = 0; countIdx < countArr.length; countIdx++) {
            while (countArr[countIdx]-- > 0) {
                arr[arrIdx++] = minVal + countIdx;
            }
        }
    }

    public static void countingSort2(int[] arr) {
        int len = arr.length;
        if (len < 2) return;
        int minVal = arr[0], maxVal = arr[0];
        for (int i = 1; i < len; i++) {
            if (arr[i] < minVal) {
                minVal = arr[i];
            } else if (arr[i] > maxVal) {
                maxVal = arr[i];
            }
        }

        int[] countArr = new int[maxVal - minVal + 1];
        for (int val : arr) {
            countArr[val - minVal]++;
        }
        for (int countIdx = countArr.length - 1, arrIdx = 0; countIdx >= 0; countIdx--) {
            while (countArr[countIdx]-- > 0) {
                arr[arrIdx++] = minVal + countIdx;
            }
        }
    }

    public static void main(String[] args) {
        int[] arr = {12, 11, 15, 50, 7, 65, 3, 99, 0};
        System.out.println("---排序前:  " + Arrays.toString(arr));
        countingSort(arr);
        System.out.println("计数排序从小到大:  " + Arrays.toString(arr));
        countingSort2(arr);
        System.out.println("计数排序从大到小:  " + Arrays.toString(arr));
    }
}

三、应用场景

  • 适用于范围较小的整数排序:计数排序适用于对范围较小的整数进行排序,因为它的时间复杂度与输入数据的范围大小线性相关,而与数据规模无关。
  • 适用于重复值较多的情况:如果输入数据中存在大量重复值,计数排序可以有效地减少比较次数,提高排序效率。
  • 稳定性:计数排序是一种稳定的排序算法,可以保持相同元素的相对顺序不变。
  • 适用于非负整数排序:计数排序要求输入数据必须是非负整数,且适用于整数排序,不适用于字符串等其他类型的数据。
  • 适用于辅助排序算法:计数排序可以作为辅助排序算法,用于优化其他排序算法的性能,例如基数排序。

参考链接:
十大经典排序算法(Java实现)

相关推荐
安逸sgr5 分钟前
SpringMVC启动流程
java·jvm·spring·spring cloud·eclipse·tomcat·maven
AndrewHZ6 分钟前
【图像处理基石】暗光增强算法入门:从原理到实战(Python+OpenCV)
图像处理·python·opencv·算法·计算机视觉·cv·暗光增强
lifallen41 分钟前
从Apache Doris 学习 HyperLogLog
java·大数据·数据仓库·算法·apache
fire-flyer42 分钟前
maven-jlink-plugin入门
java·maven
Knight_AL44 分钟前
Java 单元测试全攻略:JUnit 生命周期、覆盖率提升、自动化框架与 Mock 技术
java·junit·单元测试
cominglately1 小时前
记录一次生产环境数据库死锁的处理过程
java·死锁
用户0332126663671 小时前
在 Word 文档中插入图片的 Java 指南
java
深圳蔓延科技1 小时前
单点登录到底是什么?
java·后端
SimonKing1 小时前
除了 ${},Thymeleaf 的这些用法让你直呼内行
java·后端·程序员
智驱力人工智能1 小时前
使用手机检测的智能视觉分析技术与应用 加油站使用手机 玩手机检测
深度学习·算法·目标检测·智能手机·视觉检测·边缘计算