码迷,mamicode.com
首页 > 编程语言 > 详细

[程序员代码面试指南]数组和矩阵问题-找到无序数组中最小的k个数(堆排序)

时间:2019-04-10 00:38:37      阅读:207      评论:0      收藏:0      [点我收藏+]

标签:https   temp   超过   dma   private   组成   NPU   new   第一步   

题目链接

https://www.nowcoder.com/practice/6a296eb82cf844ca8539b57c23e6e9bf?tpId=13&tqId=11182&tPage=2&rp=2&ru=/ta/coding-interviews&qru=/ta/coding-interviews/question-ranking

题目描述

从无序序列,找到最小topk个元素。

解题思路

使用大根堆维护最小topk个元素:
- 首先前k个元素建立大根堆(从最后一个非叶节点(数组长度/2-1,结点从0计)至根节点(数组第一个元素)调整)。
- 之后维护这个最小k个元素的大根堆(比较后面的元素与根顶元素,若新元素小则替换掉堆顶元素,并进入调整)。
- 最终堆中元素即为所求。
查找topk时间复杂度:O(nlogk)。

相关知识:堆排序

堆的定义

堆是具有以下性质的完全二叉树:
每个结点的值都大于或等于其左右孩子结点的值,称为大顶堆;或者每个结点的值都小于或等于其左右孩子结点的值,称为小顶堆。

用数组表示一个堆结构,堆的定义就是:(结点从0计)
大根堆:arr[i] >= arr[2i+1] && arr[i] >= arr[2i+2]
小根堆:arr[i] <= arr[2i+1] && arr[i] <= arr[2i+2]

堆排序步骤

第一步:建堆:从最后一个非叶节点(数组长度/2-1,结点从0计)至根节点(数组第一个元素)调整。
第二步:反复执行交换、调整:将堆顶元素与树最后一个叶节点交换,从上至下调整剩余节点为堆(称为筛选);再将堆顶元素与最后一个叶节点交换...直到所有元素组成序列。大根堆对应升序,小根堆对应降序。

堆排序特点

  • 时间复杂度:平均、最好、最坏均为O(nlogn)
  • 相比快排,堆排序的最坏时间复杂度更优,这是堆排序最大的优点。所以堆排序适合记录数n较大的文件,不适合记录数较小的文件。
  • 对深度为K的堆,筛选算法关键字比较次数至多为2(K-1);则在建n个元素,深度为h的堆时,总共进行的关键字比较次数不超过4n(公式见数据结构严蔚敏P282底栏??);又,n个结点的完全二叉树深度为log2n」+1,所以最坏时间复杂度O(nlogn).

堆排序参考链接

https://www.cnblogs.com/chengxiao/p/6129630.html

代码

import java.util.*;
public class Solution {
    public ArrayList<Integer> GetLeastNumbers_Solution(int [] input, int k) {
        ArrayList<Integer> maxRootHeap = new ArrayList<Integer>();
        if(k<1||k>input.length){
            return maxRootHeap;
        }
        
        for(int i=0;i<k;++i) {
            maxRootHeap.add(input[i]);
        }
        
        //建堆
        buildMaxRootHeap(maxRootHeap);      
        
        //调整
        for(int i=k;i<input.length;++i) {
            if(input[i]<maxRootHeap.get(0)) {
                maxRootHeap.set(0, input[i]);
                heapify(maxRootHeap,0,k-1);
            }
        }
        return maxRootHeap;
    }
    
    private void buildMaxRootHeap(ArrayList<Integer> maxRootHeap) {
        for(int i=maxRootHeap.size()/2-1;i>=0;--i) {
            heapify(maxRootHeap,i,maxRootHeap.size()-1);
        }
    }
    
    private void heapify(ArrayList<Integer> maxRootHeap,int index,int heapSize) {//
        int lIdx=2*index+1;
        int rIdx=2*index+2;
        int maxIdx=index;
        while(lIdx<=heapSize) {
            if(maxRootHeap.get(lIdx)>maxRootHeap.get(index)) {
                maxIdx=lIdx;
            }
            if(rIdx<=heapSize&&maxRootHeap.get(rIdx)>maxRootHeap.get(maxIdx)) {
                maxIdx=rIdx;
            }
            
            if(maxIdx!=index) {
                swap(maxRootHeap,index,maxIdx);
            }
            else {
                break;
            }
            index=maxIdx;
            lIdx=2*index+1;
            rIdx=2*index+2;
        }           
    }
    
    private void swap(ArrayList<Integer> heap,int idx1,int idx2) {
        int temp=heap.get(idx1);
        heap.set(idx1,heap.get(idx2));
        heap.set(idx2, temp);
    }
}

[程序员代码面试指南]数组和矩阵问题-找到无序数组中最小的k个数(堆排序)

标签:https   temp   超过   dma   private   组成   NPU   new   第一步   

原文地址:https://www.cnblogs.com/coding-gaga/p/10669836.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!