博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
TopK问题
阅读量:4134 次
发布时间:2019-05-25

本文共 2719 字,大约阅读时间需要 9 分钟。

import java.util.Arrays;import java.util.PriorityQueue;import java.util.Queue;/** * 看起来分治法的快速选择算法的时间、空间复杂度都优于使用堆的方法,但是要注意到快速选择算法的几点局限性: * 第一,算法需要修改原数组,如果原数组不能修改的话,还需要拷贝一份数组,空间复杂度就上去了。 * 第二,算法需要保存所有的数据。如果把数据看成输入流的话,使用堆的方法是来一个处理一个,不需要保存数据,只需要保存 k 个元素的最大堆。 * 而快速选择的方法需要先保存下来所有的数据,再运行算法。当数据量非常大的时候,甚至内存都放不下的时候,就麻烦了。所以当数据量大的时候还是用基于堆的方法比较好。 * */public class TopK {    public int[] getLeastNumbers(int[] arr, int k) {        if(arr == null || arr.length == 0){            return new int[0];        }        if(arr.length <= k){            return arr;        }        quickSelect(arr,0,arr.length -1,k);        int[] result = new int[k];        for(int i=0;i
m,则左侧数组中的 mm 个数都属于最小的 kk 个数,我们还需要在右侧数组中寻找最小的 k-mk−m 个数,对右侧数组递归地 partition 即可。 * * 空间复杂度 O(1),不需要额外空间。 * 时间复杂度的分析方法和快速排序类似。由于快速选择只需要递归一边的数组,时间复杂度小于快速排序,期望时间复杂度为 O(n),最坏情况下的时间复杂度为 O(n^2) * * @param arr * @param start * @param end * @param k */ public void quickSelect(int[] arr,int start,int end,int k){ int pivotIndex = partition(arr,start,end); if(pivotIndex == k){ // 正好找到最小的 k(m) 个数 return ; }else if(k < pivotIndex){ // 最小的 k 个数一定在前 m 个数中,递归划分 quickSelect(arr,start,pivotIndex - 1,k); }else { // 在右侧数组中寻找最小的 k-m 个数,两次递归调用传入的参数为什么都是 k?特别是第二个调用, // 我们在右侧数组中寻找最小的 k-m 个数,但是对于整个数组而言,这是最小的 k 个数。所以说,函数调用传入的参数应该为 k。 quickSelect(arr,pivotIndex + 1,end,k); } } public int partition(int[] arr,int start,int end){ int pivot = arr[start]; int mark = start; for(int i=start + 1;i<=end;i++){ if(arr[i] < pivot){ mark ++; int temp = arr[mark]; arr[mark] = arr[i]; arr[i] = temp; } } arr[start] = arr[mark]; arr[mark] = pivot; return mark; } /** * 使用堆,这里使用优先队列 * 由于使用了一个大小为 k 的堆,空间复杂度为 O(k); * 入堆和出堆操作的时间复杂度均为 O(logk),每个元素都需要进行一次入堆操作,故算法的时间复杂度为 O(nlogk) * * @param */ public int[] getLeastNumbersByHeap(int[] arr, int k) { if(k == 0 || arr == null){ return new int[0]; } if(arr.length <= k){ return arr; } //使用一个最大堆,Java 的 PriorityQueue 默认是小顶堆,添加 comparator 参数使其变成最大堆 Queue
heap = new PriorityQueue<>(k, (i1, i2) -> Integer.compare(i2, i1)); for(int item:arr){ // 当前数字小于堆顶元素才会入堆 if(heap.isEmpty() || heap.size() < k || item < heap.peek()){ heap.offer(item); } if(heap.size() > k){ // 删除堆顶最大元素 heap.poll(); } } int[] result = new int[k]; for(int i=0;i

 

转载地址:http://cqsvi.baihongyu.com/

你可能感兴趣的文章
将一个数插入到有序的数列中,插入后的数列仍然有序
查看>>
在有序的数列中查找某数,若该数在此数列中,则输出它所在的位置,否则输出no found
查看>>
万年历
查看>>
作为码农你希望面试官当场指出你错误么?有面试官这样遭到投诉!
查看>>
好多程序员都认为写ppt是很虚的技能,可事实真的是这样么?
查看>>
如果按照代码行数发薪水会怎样?码农:我能刷到公司破产!
查看>>
程序员失误造成服务停用3小时,只得到半月辞退补偿,发帖喊冤
查看>>
码农:很多人称我“技术”,感觉这是不尊重!纠正无果后果断辞职
查看>>
php程序员看过来,这老外是在吐糟你吗?看看你中了几点!
查看>>
为什么说程序员是“培训班出来的”就是鄙视呢?
查看>>
码农吐糟同事:写代码低调点不行么?空格回车键与你有仇吗?
查看>>
阿里p8程序员四年提交6000次代码的确有功,但一次错误让人唏嘘!
查看>>
一道技术问题引起的遐想,最后得出结论技术的本质是多么的朴实!
查看>>
985硕士:非科班自学编程感觉还不如培训班出来的,硕士白读了?
查看>>
你准备写代码到多少岁?程序员们是这么回答的!
查看>>
码农:和产品对一天需求,产品经理的需求是对完了,可我代码呢?
查看>>
程序员过年回家该怎么给亲戚朋友解释自己的职业?
查看>>
技术架构师的日常工作是什么?网友:搭框架,写公共方法?
查看>>
第四章 微信飞机大战
查看>>
九度:题目1008:最短路径问题
查看>>