【剑指offer】53. 最小的k个数(java选手)(优先队列+快排+快速选择)

题目链接

acwing题目链接
力扣题目链接

题目描述

输入 n个整数,找出其中最小的 k 个数。

注意:
输出数组内元素请按从小到大顺序排序;
数据范围
1≤k≤n≤1000

样例
输入:[1,2,3,4,5,6,7,8] , k=4
输出:[1,2,3,4]

题目分析

  • 排序算法

题解

优先队列-小根堆

  • 用小根堆

将所有元素都放入小根堆中,就相当于堆元素进行了排序。
然后依次从优先队列的首部(最小的位置)开始取元素,取k个为止

注意啦!
这是优先队列(升序排列)实现的小根堆
也就是每次peek取出的元素都是当前优先队列中最小的!

class Solution {
    public List<Integer> getLeastNumbers_Solution(int[] input, int k) {
        // 优先队列(升序),小根堆,根是最小的【peek取出的元素是最小的】
        PriorityQueue<Integer> pq = new PriorityQueue<>((a, b)->a-b);
        for(int i = 0; i < input.length; i ++){
            pq.add(input[i]);
        }
        
        List<Integer> list = new ArrayList<>();
        for(int i = 0; i < k; i ++){
            list.add(pq.poll());
        }
  
        return list;
    }
}

优先队列-大根堆

  • 从大到小排序,首部元素是最大的
  • 只存放k个整数,如果超过这个值之后,遍历的元素与首部元素小才插入(因为要找前k个最小的元素)
  • 遍历优先队列,不停的在list首部插入值(因为是大根堆,所以队列首部的元素是最大的)
class Solution {
    public List<Integer> getLeastNumbers_Solution(int[] input, int k) {
        // 大根堆,堆中只存放k个整数,超过这个k值后,如果元素比peek得到的值小,插入
        // 优先队列从大到小排序,逆序
        PriorityQueue<Integer> pq = new PriorityQueue<>((a, b)-> b - a);
        
        for(int i = 0; i < input.length; i ++){
            if(pq.size() >= k){
                if(pq.peek() > input[i]){
                    pq.poll();
                    pq.add(input[i]);
                }
            }else{
                pq.add(input[i]);
            }
        }
        
        List<Integer> list = new ArrayList<>();
        for(int i = 0; i < k; i ++){
            list.add(0, pq.poll());
        }
        
        return list;
    }
}

!0323更新!上述代码在力扣上无法通过,没有考虑到k为0的情况;下面是更新版本

class Solution {
    public int[] smallestK(int[] arr, int k) {
        if(k==0) return new int[0];
        // 优先队列,降序
        PriorityQueue<Integer> pq = new PriorityQueue<>((a,b)-> b-a);
        for(int i = 0; i < arr.length; i ++){
            if(pq.size() >= k){
                if(arr[i] < pq.peek()){
                    pq.poll();
                    pq.add(arr[i]);
                }  
            }else{
                pq.add(arr[i]);
            }
        }
        int[] res = new int[k];
        for(int i = 0; i < k; i ++){
            res[i] = pq.poll();
        }
        return res;
    }
}

快排

写法1:

class Solution {
    public int[] smallestK(int[] arr, int k) {
        sort(arr, 0, arr.length - 1);
        int[] res = new int[k];
        for(int i = 0; i < k; i ++){
            res[i] = arr[i];
        }
        return res;
    }

    public void sort(int[] arr, int left, int right){
        // 只有最后一个元素
        if(left >= right) return;

        // 基准
        int base = arr[left];
        int i = left, j = right;
        while(i < j){
            while(i < j && arr[j] > base){
                j--;
            }
            if(i < j){
                arr[i] = arr[j];
                i ++;
            }


            while(i < j && arr[i] < base){
                i ++;
            }
            if(i < j){
                arr[j] = arr[i];
                j --;
            }
        }

        arr[i] = base;
        sort(arr, left, i - 1);
        sort(arr, i + 1, right);
    }
}

写法2:

class Solution {
    public int[] smallestK(int[] arr, int k) {
        sort(arr, 0, arr.length - 1);
        int[] res = new int[k];
        for(int i = 0; i < k; i ++){
            res[i] = arr[i];
        }
        return res;
    }

    public void sort(int[] arr, int left, int right){
        // 只有最后一个元素
        if(left >= right) return;

        // 基准
        int base = arr[(left+right)/2];
        int i = left-1, j = right+1;
        while(i < j){
           do i++; while(arr[i] < base);
           do j--; while(arr[j] > base);
           if(i < j){
            int temp = arr[i];
            arr[i] = arr[j];
            arr[j] = temp;
           }
        }
        sort(arr, left, j);
        sort(arr, j + 1, right);
    }
}

快速选择

题目链接;找第k个元素

class Solution {
    public int findKthLargest(int[] nums, int k) {
        // 需要数组中下标为need的元素
        int need = nums.length - k;
        return sort(nums, 0, nums.length - 1, need);
    }

    public int sort(int[] nums, int left, int right, int k) {
        if(left == right) return nums[left];

        int base = nums[(left + right) / 2];
        int i = left - 1, j = right + 1;
        while (i < j) {
            do i++; while (nums[i] < base);
            do j--; while (nums[j] > base);
            if (i < j) {
                int temp = nums[i];
                nums[i] = nums[j];
                nums[j] = temp;
            }
        }

        if(j >= k) return sort(nums, left, j, k);
        else return sort(nums, j + 1, right, k);
    }
}

注意!边界

if(j >= k) return sort(nums, left, j, k);
else return sort(nums, j + 1, right, k);
  • 14
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值