计数排序 Leetcode 面试经典150-H指数

java 复制代码
public class Solution {
    public int hIndex(int[] citations) {
        int n = citations.length, tot = 0;
        int[] counter = new int[n + 1];
        for (int i = 0; i < n; i++) {
            if (citations[i] >= n) {
                counter[n]++;
            } else {
                counter[citations[i]]++;
            }
        }
        for (int i = n; i >= 0; i--) {
            tot += counter[i];
            if (tot >= i) {
                return i;
            }
        }
        return 0;
    }
}

计数排序(Counting Sort)是一种非比较型的排序算法,适用于一定范围内的整数排序。它的核心思想是将输入的数据值转化为键存储在额外开辟的数组空间中,从而达到排序的目的。计数排序的时间复杂度为 O(n + k),其中 n 是输入数组的长度,k 是输入数据的范围。

计数排序的基本步骤:

  1. 确定数据范围

    • 找出输入数组中的最大值和最小值,确定数据的范围。
  2. 初始化计数数组

    • 创建一个大小为 k(数据范围大小)的数组 count,用来存储每个数值出现的次数。
  3. 统计每个数值出现的次数

    • 遍历输入数组,对于数组中的每一个元素,将其值作为索引,在 count 数组中相应的位置上累加 1。
  4. 构建输出数组

    • 创建一个新的输出数组 output,用来存放排序后的结果。
    • 再次遍历 count 数组,对于 count 中的每个非零元素,将其对应的索引值复制到 output 数组中相应次数。
  5. 累计计数数组(可选):

    • 如果需要构建累积计数数组,可以将 count 数组中的元素进行累积求和,这样可以知道每个数值在最终排序数组中的起始位置。
  6. 填充输出数组

    • 根据累积计数数组,从输入数组中依次填入输出数组,保证稳定性(即相同的数值保持原有的顺序)。

示例代码:

假设我们要对一个整数数组进行排序,且已知这些整数的范围是从 minValuemaxValue

复制代码
public void countingSort(int[] array, int minValue, int maxValue) {
    int size = maxValue - minValue + 1;
    int[] count = new int[size];

    // Step 2: Count each element's frequency
    for (int number : array) {
        count[number - minValue]++;
    }

    // Step 4: Build the output array
    int index = 0;
    for (int i = 0; i < size; i++) {
        while (count[i] > 0) {
            array[index++] = i + minValue;
            count[i]--;
        }
    }
}

适用场景:

  • 当输入数据的范围不是很大时(即 k 不太大),计数排序是非常高效的。
  • 特别适用于数据分布均匀的情况。

缺点:

  • 当输入数据的范围很大时,需要开辟较大的额外空间,可能导致空间浪费。
  • 只适用于整数排序,且适用于非负整数排序最为理想。

计数排序的应用:

  • 计数排序常用于预处理阶段,比如在基数排序(Radix Sort)中作为子步骤。
  • 在处理特定范围内的数据时非常有效,例如处理考试成绩、年龄等数据。

通过这种方式,计数排序能够在 O(n + k) 时间复杂度内完成排序任务,特别适合数据范围有限的情况。对于 H-index 的问题,计数排序可以有效地利用引用次数的范围来快速解决问题。

相关推荐
LYFlied5 小时前
【每日算法】LeetCode 105. 从前序与中序遍历序列构造二叉树
数据结构·算法·leetcode·面试·职场和发展
重生之我是Java开发战士5 小时前
【数据结构】Java对象的比较
java·jvm·数据结构
DanyHope5 小时前
LeetCode 206. 反转链表:迭代 + 递归双解法全解析
算法·leetcode·链表·递归·迭代
NAGNIP5 小时前
才发现TensorBoard是个可视化的神器!
算法
历程里程碑5 小时前
C++ 16:C++11新特化
c语言·开发语言·数据结构·c++·经验分享
_dindong5 小时前
算法杂谈:回溯路线
数据结构·算法·动态规划·bfs·宽度优先
咋吃都不胖lyh5 小时前
详解 UCB 算法的置信区间与核心逻辑(通俗 + 公式 + 实例)
人工智能·算法·机器学习
DanyHope6 小时前
LeetCode 两数之和:从 O (n²) 到 O (n),空间换时间的经典实践
前端·javascript·算法·leetcode·职场和发展
free-elcmacom6 小时前
机器学习高阶教程<6>推荐系统高阶修炼手册:混排、多任务与在线学习,解锁精准推荐新境界
人工智能·python·学习·算法·机器学习·机器人
断剑zou天涯6 小时前
【算法笔记】AC自动机
java·笔记·算法