如何用TopN算法在10亿个整数中找出前1000个最大的数

技术如何用TopN算法在10亿个整数中找出前1000个最大的数如何用TopN算法在10亿个整数中找出前1000个最大的数,相信很多没有经验的人对此束手无策,为此本文总结了问题出现的原因和解决方法,通过这篇文章希望你能解决

如何用TopN算法找出10亿个整数中最大的前1000个数字,相信很多没有经验的人对此无能为力。因此,本文总结了出现问题的原因和解决方法,希望大家可以通过这篇文章来解决这个问题。

面试题目:如何在10亿个整数中找出前1000个最大的数字。

我们知道有很多排序算法:

冒泡算法:通过两层for循环,外循环找到数组中最大的元素,第一次放在倒数第二的位置,第二次循环找到第二大的元素,放在倒数第二的位置。循环n次找到TopN。

缺点:冒泡排序内循环需要大量的交换元素。复杂性介于0(n)和0(N2)之间。

快速排序:选择一个基准元素,在每次排序中可以放在正确的位置。左边的元素比基准小,右边的元素比基准大,这样数组就可以分为左右两部分。TopN问题也是如此。选择一个基准元素,并通过快速排序将其放在正确的位置。如果左边的元素数量少于1000,那么从基准的右边继续排序。如果左边的元素数量超过1000,那么从基准的左边开始排序,直到基准的位置正好是1000。

缺点:第一排序复杂度为O(n),第二排序复杂度为O(n/2),第三排序复杂度为O(n/4)。

文件存储,分而治之:

小于基准的元素存储在txt1中,大于基准的文件存储在txt2中。然后,以类似于方法2的方式,最终获得TopN。

缺点:磁盘读写次数太多。

MapReduce:的内存和性能确实有限,所以我们可以在不同的机器上存储10亿段,每台机器计算自己的TopN,最后总结出来。

缺点:以空间换时间。

00-1010,长度为n的数组保存在内存中。根据堆的性质,每个节点都小于它的左右子节点。首先,取出前n个数字,构建一个小的顶部堆,然后将所有数据与堆顶部进行比较。如果小于堆顶,则直接丢弃;如果它更大,它将替换堆顶部并重建堆。

构建小顶堆的过程:首先找到最后一个非叶节点,数组的长度是6,然后最后一个非叶节点是长度/2-1,也就是6/2-1=2,然后下一步是比较这个节点的值和它的左右节点值,如果这个节点的值大于它的左\右子树的值(意思是把最小值放在这个节点中)就交换。如果该节点不是叶节点,则重复该过程,直到该节点成为叶节点。

具体执行代码如下:

import Java . util . random;

/**

*@authorvincent

*@time2019-08-0711:59

*/

publicclassTopN{

publicationstatintn=10;//Top10

publicationstatintlen=100000000;//1亿个整数。

publicationstatintarrs[]=new int[LEN];

publicationstationresult[]=new int[N];//在内存中维护一个长度为n的小顶堆。

publicationstatintlen=result . length;

//堆中元素的有效元素heapSize=len。

publicationstatinthesize=len;

publicationstativitmain(String[]args){ 0

//生成随机数组。

for(inti=0;iLENI){ 0

sp;       arrs[i] = new Random().nextInt(999999999);
        }
        //构建初始堆
        for(int i =  0;i<N;i++){
            result[i] = arrs[i];
        }
        //构建小顶堆
        long start =System.currentTimeMillis();
        buildMinHeap();
        for(int i = N;i<LEN;i++){
            if(arrs[i] > result[0]){
                result[0] = arrs[i];
                minHeap(0);
            }
        }
        System.out.println(LEN+"个数,求Top"+N+",耗时"+(System.currentTimeMillis()-start)+"毫秒");
        print();
    }
    /**
     * 自底向上构建小堆
     */
    public static void buildMinHeap(){
        int size = len / 2 -1 ; //最后一个非叶子节点
        for(int i = size;i>=0;i--){
            minHeap(i);
        }
    }
    /**
     * i节点为根及子树是一个小堆
     * @param i
     */
    public static void minHeap(int i){
        int l = left(i);
        int r = right(i);
        int index = i;
        if(l<heapSize && result[l]< result[index]){
            index = l;
        }
        if(r<heapSize && result[r]< result[index]){
            index = r;
        }
        if(index != i){
            int t = result[index];
            result[index] = result[i];
            result[i] = t;
            //递归向下构建堆
            minHeap(index);
        }
    }
    /**
     * 返回i节点的左孩子
     * @param i
     * @return
     */
    public static int left(int i){
        return 2*i;
    }
    /**
     * 返回i节点的右孩子
     * @param i
     * @return
     */
    public static int right(int i){
        return 2*i+1;
    }
    /**
     * 打印
     */
    public  static void print(){
        for(int a: result){
            System.out.print(a+",");
        }
        System.out.println();
    }
}

看完上述内容,你们掌握如何用TopN算法在10亿个整数中找出前1000个最大的数的方法了吗?如果还想学到更多技能或想了解更多相关内容,欢迎关注行业资讯频道,感谢各位的阅读!

内容来源网络,如有侵权,联系删除,本文地址:https://www.230890.com/zhan/39237.html

(0)

相关推荐

  • MySQL慢SQL采集方案分析

    技术MySQL慢SQL采集方案分析本篇内容主要讲解“MySQL慢SQL采集方案分析”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“MySQL慢SQL采集方案分析”吧!作为一名M

    攻略 2021年11月5日
  • 手写Compose方法的有哪些

    技术手写Compose方法的有哪些本篇内容主要讲解“手写Compose方法的有哪些”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“手写Compose方法的有哪些”吧!前言:为什

    攻略 2021年10月21日
  • linux上的并行软件ParaFly怎么用

    技术linux上的并行软件ParaFly怎么用这篇文章将为大家详细讲解有关linux上的并行软件ParaFly怎么用,文章内容质量较高,因此小编分享给大家做个参考,希望大家阅读完这篇文章后对相关知识有一定的了解。

    攻略 2021年12月10日
  • BeanDefinition的原理是什么

    技术BeanDefinition的原理是什么这期内容当中小编将会给大家带来有关BeanDefinition的原理是什么,文章内容丰富且以专业的角度为大家分析和叙述,阅读完这篇文章希望大家可以有所收获。一、BeanDefi

    攻略 2021年12月3日
  • 怎么理解java图的对象化描述

    技术怎么理解java图的对象化描述这篇文章主要讲解了“怎么理解java图的对象化描述”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“怎么理解java图的对象化描述”吧!一、

    攻略 2021年11月9日
  • c语言圣诞树(c语言简单圣诞树编程)

    技术如何用C语言实现圣诞树这篇文章主要介绍“如何用C语言实现圣诞树”,在日常操作中,相信很多人在如何用C语言实现圣诞树问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”如何用C语言实现圣诞树”

    攻略 2021年12月22日