每日一题--最小的k个数

两种解法:
一种基于Partition(将要改变输入数组),时间复杂度O(n)
另一种使用最大堆的方法(可以解决海量数据),时间复杂度O(nlogk)

重点:最大堆的建立方法(从下往上O(logn))


//Key[i]>=Key[2i+1]&&key>=key[2i+2]称为大顶堆
//建最大堆
import java.util.Arrays;

/**
 * 问题描述:输入n个整数,输出其最小的k个数
 * 输入:4,5,1,6,2,7,3,8
 * 输出:1,2,3,4
 * Created by lxq on 2017/9/3.
 */
public class Problem2 {
    public static void main(String[] args){
        Problem2 test = new Problem2();
        int[] array = {4,5,1,6,2,7,3,8};
        test.getLeastNumbers(array,2);
    }
    public void getLeastNumbers(int[] array,int k ){
        if(array==null||k<0||k>array.length)
            return;
        int[] kArray = Arrays.copyOfRange(array,0,k);
        buildMaxHeap(kArray);
        //整个数组的 最大元素 总是在最大堆的第一个,即kArray[0]
        for(int i=k;i<array.length;i++){
            if(array[i]<kArray[0]){
                kArray[0] = array[i];
                //替换后需要再次维护最大堆
                maxHeap(kArray,0);
            }
        }
        for(int i :kArray)
            System.out.println(i);
    }

    private void maxHeap(int[] kArray, int i) {
        int left = 2*i+1;
        int right = 2*i+2;
        int largest = 0;
        if(right<kArray.length&&kArray[left]>kArray[i]){
            largest = left;
        }else{
            largest = i;
        }
        if(right<kArray.length&&kArray[right]>kArray[largest]){
            largest = right;
        }
        if(largest!=i){
            int temp = kArray[i];
            kArray[i] = kArray[largest];
            kArray[largest] = temp;
            //重点!!根节点换成laregst可能会破坏最大堆结构
            maxHeap(kArray,largest);
        }

    }

    //建最大堆
    private void buildMaxHeap(int[] kArray) {
        for(int i=kArray.length/2;i>=0;i--){
            maxHeap(kArray,i);
        }
    }
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值