题目描述(中等难度)

找出第 k 大的数。

解法一 暴力

使用快排从大到小排序,将第 k 个数返回即可。

我们直接使用 java 提供的排序算法,又因为默认是从小到大排序,所以将倒数第 k 个数返回即可。

public int findKthLargest(int[] nums, int k) {
    Arrays.sort(nums);
    return nums[nums.length - k];
}

解法二

我们没必要把所有数字正确排序,我们可以借鉴快排中分区的思想,这里不细讲了,大家可以去回顾一下快排。

随机选择一个分区点,左边都是大于分区点的数,右边都是小于分区点的数。左部分的个数记做 m

如果 k == m + 1,我们把分区点返回即可。

如果 k > m + 1,说明第 k 大数在右边,我们在右边去寻找第 k - m - 1 大的数即可。

如果 k < m + 1,说明第 k 大数在左边,我们在左边去寻找第 k 大的数即可。

左边和右边寻找在代码中采取递归即可。

分区达到的效果就是下边的样子。

原数组 3 7 6 1 5

如果把 5 作为分区点,那么数组最后就会变成下边的样子, i 指向最终的分区点
7 6 5 1 3
    ^
    i

代码的话,分区可以采取双指针,i 前边始终存比分区点大的元素。

public int findKthLargest(int[] nums, int k) {
    return findKthLargestHelper(nums, 0, nums.length - 1, k);
}

private int findKthLargestHelper(int[] nums, int start, int end, int k) {
    int i = start;
    int pivot = nums[end];//分区点
    //将 i 的左半部分存比分区点大的数
    //将 i 的右半部分存比分区点小的数
    for (int j = start; j < end; j++) {
        if (nums[j] >= pivot) {
            int temp = nums[i];
            nums[i] = nums[j];
            nums[j] = temp;
            i++;
        }
    }
    //分区点放到 i 的位置
    int temp = nums[i];
    nums[i] = pivot;
    nums[end] = temp;
    //左边的数量加上 1
    int count = i - start + 1;
    if (count == k) {
        return nums[i];
    //从右边去继续寻找
    } else if (count < k) {
        return findKthLargestHelper(nums, i + 1, end, k - count);
    //从左边去继续寻找    
    } else {
        return findKthLargestHelper(nums, start, i - 1, k);
    }
}

解法三

我们可以使用优先队列,建一个最大堆,然后依次弹出元素,弹出的第 k 个元素就是我们要找的。

优先队列的使用也不是第一次了,之前在 23 题188 题 也用过,原理可以参考 这里 这里

这里我们直接使用 java 提供的优先队列了。

public int findKthLargest(int[] nums, int k) {
    Comparator<Integer> cmp;
    cmp = new Comparator<Integer>() {
        @Override
        public int compare(Integer i1, Integer i2) {
            // TODO Auto-generated method stub
            return i2 - i1;
        }
    };

    // 建立最大堆
    Queue<Integer> q = new PriorityQueue<Integer>(cmp);
    for (int i = 0; i < nums.length; i++) {
        q.offer(nums[i]);
    }

    for (int i = 0; i < k - 1; i++) {
        q.poll();
    }
    return q.poll();
}

java 默认的是建最小堆,所以我们需要一个比较器来改变优先级。

如果使用最小堆也可以解决这个问题,只需要保证队列中一直是 k 个元素即可。当队列超出 k 个元素后,把队列中最小的去掉即可,这就保证了最后队列中的元素一定是前 k 大的元素。

public int findKthLargest(int[] nums, int k) {
    // 建立最小堆
    Queue<Integer> q = new PriorityQueue<Integer>();
    for (int i = 0; i < nums.length; i++) {
        q.offer(nums[i]);
        if (q.size() > k) {
            q.poll();
        }
    }
    return q.poll();
}

这道题不是很难,只要掌握了快排的思想,解法二也能很快写出来。解法三的话,就得事先了解优先队列了。

results matching ""

    No results matching ""