热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

No5、查找最小的k个元素(数组)

题目:输入n个整数,输出其中最小的k个。例如输入1,2,3,4,5,6,7和8这8个数字,则最小的4个数字为1,2,3和4。编程之美原题,只不过编程
题目:输入 n 个整数,输出其中最小的 k 个。

例如输入 1,2,3,4,5,6,7 和 8 这 8 个数字,则最小的 4 个数字为 1,2,3 和 4。


编程之美原题,只不过编程之美上是求n个整数的k个最大的数


1、拿到题目第一个想法是排序,排序完了之后再取前K个最小的,这样的时间复杂度是O(n*lgn)+O(k) = O(n*lgn)...当然..这个不行,但是我们可以从最简单的方法向外延伸


2、观察上面的排序,是将n个数全部排序,其实我们不需要知道所有的顺序,我们只需要知道最小的k个数。所以,采取部分选择排序,每次遍历寻找最小的数,然后取出,遍历k次就能获得最终结果,这样的时间复杂度是O(n*k),当k


3、第二种方法取出的k个数是有顺序的,而题目不需要有顺序,所以我们需要将把求顺序的时间去掉,还能进一步优化。再继续思考,我们所求的k个数是最小的,相当于将数组分为三部分,A-n-B,A的长度为k,n左边的数都比n小,n右边的数都比n大.....看到这里是不是很熟悉,对了,快排...只能说这是一个部分快排算法...

快排的算法如下:

public void QuickSort(int[] array,int head,int tail)
{
if(head >= tail)
return;

int low = head;
int high = tail;
int num = array[low];

while(low {
while(array[high] >= num && low high--;
array[low] = array[high];

while(array[low]<=num && low low++;
array[high] = array[low];
}

array[low] = num;

QuickSort(array,head,low-1);
QuickSort(array,low+1,tail);
}
快排是取范围内的第一个数作为中枢,遍历完后数组中中枢数左边的是比中枢数小的数,中枢数右边是比中枢数大的数。题目中还要求了k个数字的限制,所以,我们在递归的时候需要对进入递归的条件做一下处理:

      假设中枢数最后的位置是index,范围的开头是head,结尾时tail,那么我们考虑的是head~index之间的数:

      1、如果head~index中数字的个数大于k,那么所有的k个最小的数都在左边区域中,所以左边子数组进入递归;

      2、如果head~index中数字的个数等于k,那么输出左边区域的数字;

      3、如果head~index中数字的个数小于k,那么最小的k个数分成了两个部分,一部分是head~index中的全部的数字,一部分是index~tail中的最小的k-len(head~index)个数字...使用一个memory保存第一部分,将第二部分进入递归即可

代码如下:

	public void getMinK(int[] list,int index)
{
if(index == 0 || list.length == 0)
return;

int pos = 0;
int num = list[0];
int low = 0;
int high = list.length-1;
//快排分区域
while(low {
while(list[high] >= num && high>low)
high--;
list[pos] = list[high];
pos = high;
list[pos] = num;

while(list[low] <= num && high >low)
low++;
list[pos] = list[low];
pos = low;
list[pos] = num;
}
//实现的时候,每次进入递归的都是新建的一个数组,相当于每次的head都是0
if(pos+1>index) //当左边区域的个数大于k
{
int[] a = new int[pos]; //新建一个数组,里面包含左边区域的个数,然后在这个数组里面挑选最小的k个数子
for(int i = 0;i {
a[i] = list[i];
}
getMinK(a, index);
}

else if(pos+1 == index) //直接输出
{
String result = "";
for(int i = 0;i<=pos;i++)
{
result = result + " " + list[i];
}
System.out.println(result);
}
else //当左边区域的个数小于k
{
String result = ""; //输出左边区域的数字
for(int i = 0;i<=pos;i++)
{
result = result + " " + list[i];
}
System.out.println(result);

int[] a = new int[list.length-pos-1]; //新建一个数组,里面存储了右边区域的数字,然后从里面取k-len(左边区域)个最小值
for(int i = 0;i {
a[i] = list[i+pos+1];
}
getMinK(a, index-pos-1);
}
}

这样的平均时间复杂度是O(N*lgk),计算方法与快排的时间复杂度的计算方法类似


4、前面的实现貌似有点麻烦..就这么着吧..重点在于思想...用方法3的前提是有一个数组能存n个数据,如果n非常大的话数据就不能都装入内存中,所以我们需要考虑其他的方法了...n太大,我们可以从k入手。在内存中维持一个容量为k的桶,维持桶中数字的最大值。先将前K个数放进桶中,依次将剩下的n-k个数尝试放进桶中。如果新的数字小于桶的最大值,那么需要将桶的最大值拿出桶外,将新的数字放入桶中,并更新桶中数字的最大值..相当于时刻保证桶中的数字永远是目前涉及到的数字中的最小值.有点动态规划的意思....

这个方法里面需要考虑一个问题,每次桶中的数字更新的话都需要获得桶中的最大值,遍历的时间复杂度是o(k),遍历永远是最笨的方法...每次都是最大值,什么算法能快速的维持一个数组的最大值,很明显是最大堆..我们将桶内部的存储格式设置成一个最大堆,堆顶的数字永远是堆中最大的数字。如果堆更新,重建堆的时间复杂度是O(lgk),是要优于O(k)的...所以我们可以用堆排序的思想来解答这个问题..

堆排序算法:

public class HeapSort {
public static void HeapAdjust(int[] array,int start,int end)
{
for(int i = start * 2;i <= end;i = i * 2)
{
if(i array[i])
i++;

if(array[start] >= array[i])
break;

int temp = array[i];
array[i] = array[start];
array[start] = temp;

start = i;
}

}
public static void main(String[] args) {
int[] array = {-1,3,4,43,23,4,43,5,34,35,3,4,43,53,3,42,3};
int length = array.length-1;

for(int i = length/2;i>0;i--)
HeapAdjust(array, i, length);

for(int i = length;i>0;i--)
{
System.out.println(array[1]);
array[1] = array[i];

HeapAdjust(array, 1, i);
}
}
}


我们只要建立个堆,然后每次插入数据后整理这个最大堆,就能得到最后结果了...代码不写了..

时间复杂度也是O(Nlgk)



推荐阅读
  • Iamtryingtomakeaclassthatwillreadatextfileofnamesintoanarray,thenreturnthatarra ... [详细]
  • 本文介绍了使用Java实现大数乘法的分治算法,包括输入数据的处理、普通大数乘法的结果和Karatsuba大数乘法的结果。通过改变long类型可以适应不同范围的大数乘法计算。 ... [详细]
  • Java容器中的compareto方法排序原理解析
    本文从源码解析Java容器中的compareto方法的排序原理,讲解了在使用数组存储数据时的限制以及存储效率的问题。同时提到了Redis的五大数据结构和list、set等知识点,回忆了作者大学时代的Java学习经历。文章以作者做的思维导图作为目录,展示了整个讲解过程。 ... [详细]
  • 本文详细介绍了Java中vector的使用方法和相关知识,包括vector类的功能、构造方法和使用注意事项。通过使用vector类,可以方便地实现动态数组的功能,并且可以随意插入不同类型的对象,进行查找、插入和删除操作。这篇文章对于需要频繁进行查找、插入和删除操作的情况下,使用vector类是一个很好的选择。 ... [详细]
  • Java太阳系小游戏分析和源码详解
    本文介绍了一个基于Java的太阳系小游戏的分析和源码详解。通过对面向对象的知识的学习和实践,作者实现了太阳系各行星绕太阳转的效果。文章详细介绍了游戏的设计思路和源码结构,包括工具类、常量、图片加载、面板等。通过这个小游戏的制作,读者可以巩固和应用所学的知识,如类的继承、方法的重载与重写、多态和封装等。 ... [详细]
  • 本文介绍了如何在给定的有序字符序列中插入新字符,并保持序列的有序性。通过示例代码演示了插入过程,以及插入后的字符序列。 ... [详细]
  • JavaSE笔试题-接口、抽象类、多态等问题解答
    本文解答了JavaSE笔试题中关于接口、抽象类、多态等问题。包括Math类的取整数方法、接口是否可继承、抽象类是否可实现接口、抽象类是否可继承具体类、抽象类中是否可以有静态main方法等问题。同时介绍了面向对象的特征,以及Java中实现多态的机制。 ... [详细]
  • 本文讨论了一个关于cuowu类的问题,作者在使用cuowu类时遇到了错误提示和使用AdjustmentListener的问题。文章提供了16个解决方案,并给出了两个可能导致错误的原因。 ... [详细]
  • C# 7.0 新特性:基于Tuple的“多”返回值方法
    本文介绍了C# 7.0中基于Tuple的“多”返回值方法的使用。通过对C# 6.0及更早版本的做法进行回顾,提出了问题:如何使一个方法可返回多个返回值。然后详细介绍了C# 7.0中使用Tuple的写法,并给出了示例代码。最后,总结了该新特性的优点。 ... [详细]
  • 本文探讨了C语言中指针的应用与价值,指针在C语言中具有灵活性和可变性,通过指针可以操作系统内存和控制外部I/O端口。文章介绍了指针变量和指针的指向变量的含义和用法,以及判断变量数据类型和指向变量或成员变量的类型的方法。还讨论了指针访问数组元素和下标法数组元素的等价关系,以及指针作为函数参数可以改变主调函数变量的值的特点。此外,文章还提到了指针在动态存储分配、链表创建和相关操作中的应用,以及类成员指针与外部变量的区分方法。通过本文的阐述,读者可以更好地理解和应用C语言中的指针。 ... [详细]
  • 从零学Java(10)之方法详解,喷打野你真的没我6!
    本文介绍了从零学Java系列中的第10篇文章,详解了Java中的方法。同时讨论了打野过程中喷打野的影响,以及金色打野刀对经济的增加和线上队友经济的影响。指出喷打野会导致线上经济的消减和影响队伍的团结。 ... [详细]
  • 猜字母游戏
    猜字母游戏猜字母游戏——设计数据结构猜字母游戏——设计程序结构猜字母游戏——实现字母生成方法猜字母游戏——实现字母检测方法猜字母游戏——实现主方法1猜字母游戏——设计数据结构1.1 ... [详细]
  • [大整数乘法] java代码实现
    本文介绍了使用java代码实现大整数乘法的过程,同时也涉及到大整数加法和大整数减法的计算方法。通过分治算法来提高计算效率,并对算法的时间复杂度进行了研究。详细代码实现请参考文章链接。 ... [详细]
  • Go语言实现堆排序的详细教程
    本文主要介绍了Go语言实现堆排序的详细教程,包括大根堆的定义和完全二叉树的概念。通过图解和算法描述,详细介绍了堆排序的实现过程。堆排序是一种效率很高的排序算法,时间复杂度为O(nlgn)。阅读本文大约需要15分钟。 ... [详细]
  • Spring特性实现接口多类的动态调用详解
    本文详细介绍了如何使用Spring特性实现接口多类的动态调用。通过对Spring IoC容器的基础类BeanFactory和ApplicationContext的介绍,以及getBeansOfType方法的应用,解决了在实际工作中遇到的接口及多个实现类的问题。同时,文章还提到了SPI使用的不便之处,并介绍了借助ApplicationContext实现需求的方法。阅读本文,你将了解到Spring特性的实现原理和实际应用方式。 ... [详细]
author-avatar
执子之手2502891083
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有