[剑指offer][JAVA][面试第40题][最小的k个数][快选][堆][BST]

【问题描述】面试第40题 最小的k个数

输入整数数组 arr ,找出其中最小的 k 个数。例如,输入4、5、1、6、2、7、3、8这8个数字,则最小的4个数字是1、2、3、4。
示例 :
输入:arr = [3,2,1], k = 2
输出:[1,2] 或者 [2,1]
0 <= k <= arr.length <= 10000
0 <= arr[i] <= 10000

【解答思路】

1. Arrays.sort
  • 需要先把该数组排序(从小到大)。
  • 截取前k个数返回需要的数组。
    时间复杂度:O(N^2) 空间复杂度:O(N)
class Solution {public int[] getLeastNumbers(int[] arr, int k) {Arrays.sort(arr);int[] a = new int[k] ;/*for(int j = 0; j <  k ; j++){a[j] = arr[j];}*/System.arraycopy(arr, 0, a, 0, k);return a;//return Arrays.copyOfRange(arr, 0, k);}
}
2. 堆

使用堆数据结构来辅助得到最小的 k 个数。堆的性质是每次可以找出最大或最小的元素。我们可以使用一个大小为 k 的最大堆(大顶堆),将数组中的元素依次入堆,当堆的大小超过 k 时,便将多出的元素从堆顶弹

时间复杂度:O(Nlogk) 空间复杂度:O(k)
使用了一个大小为 k 的堆

public int[] getLeastNumbers(int[] arr, int k) {if (k == 0) {return new int[0];}// 使用一个最大堆(大顶堆)// Java 的 PriorityQueue 默认是小顶堆,添加 comparator 参数使其变成最大堆Queue<Integer> heap = new PriorityQueue<>(k, (i1, i2) -> Integer.compare(i2, i1));for (int e : arr) {// 当前数字小于堆顶元素才会入堆if (heap.isEmpty() || heap.size() < k || e < heap.peek()) {heap.offer(e);}if (heap.size() > k) {heap.poll(); // 删除堆顶最大元素}}// 将堆中的元素存入数组int[] res = new int[heap.size()];int j = 0;for (int e : heap) {res[j++] = e;}return res;
}作者:nettee
链接:https://leetcode-cn.com/problems/zui-xiao-de-kge-shu-lcof/solution/tu-jie-top-k-wen-ti-de-liang-chong-jie-fa-you-lie-/
3. 快排变形
  • “查找第 k 大的元素”是一类算法问题,称为选择问题。
  • 找第 k 大的数,或者找前 k 大的数,有一个经典的 quick select(快速选择)算法。
  • 这个名字和 quick sort(快速排序)很像,算法的思想也和快速排序类似,都是分治法的思想

partition
时间复杂度:期望O(N) 最坏O(N^2) 空间复杂度:O(1)

public int[] getLeastNumbers(int[] arr, int k) {if (k == 0) {return new int[0];} else if (arr.length <= k) {return arr;}// 原地不断划分数组partitionArray(arr, 0, arr.length - 1, k);// 数组的前 k 个数此时就是最小的 k 个数,将其存入结果int[] res = new int[k];for (int i = 0; i < k; i++) {res[i] = arr[i];}return res;
}void partitionArray(int[] arr, int lo, int hi, int k) {// 做一次 partition 操作int m = partition(arr, lo, hi);// 此时数组前 m 个数,就是最小的 m 个数if (k == m) {// 正好找到最小的 k(m) 个数return;} else if (k < m) {// 最小的 k 个数一定在前 m 个数中,递归划分partitionArray(arr, lo, m-1, k);} else {// 在右侧数组中寻找最小的 k-m 个数partitionArray(arr, m+1, hi, k);}
}// partition 函数和快速排序中相同,具体可参考快速排序相关的资料
// 代码参考 Sedgewick 的《算法4》
int partition(int[] a, int lo, int hi) {int i = lo;int j = hi + 1;int v = a[lo];while (true) { while (a[++i] < v) {if (i == hi) {break;}}while (a[--j] > v) {if (j == lo) {break;}}if (i >= j) {break;}swap(a, i, j);}swap(a, lo, j);// a[lo .. j-1] <= a[j] <= a[j+1 .. hi]return j;
}void swap(int[] a, int i, int j) {int temp = a[i];a[i] = a[j];a[j] = temp;
}作者:nettee
链接:https://leetcode-cn.com/problems/zui-xiao-de-kge-shu-lcof/solution/tu-jie-top-k-wen-ti-de-liang-chong-jie-fa-you-lie-/
4. 计数排序

时间复杂度:O(N) 空间复杂度:O(N)

class Solution {public int[] getLeastNumbers(int[] arr, int k) {if (k == 0 || arr.length == 0) {return new int[0];}// 统计每个数字出现的次数int[] counter = new int[10001];for (int num: arr) {counter[num]++;}// 根据counter数组从头找出k个数作为返回结果int[] res = new int[k];int idx = 0;for (int num = 0; num < counter.length; num++) {while (counter[num]-- > 0 && idx < k) {res[idx++] = num;}if (idx == k) {break;}}return res;}
}作者:sweetiee
链接:https://leetcode-cn.com/problems/zui-xiao-de-kge-shu-lcof/solution/3chong-jie-fa-miao-sha-topkkuai-pai-dui-er-cha-sou/
5. 二叉搜索树BST(前K大有序)

class Solution {public int[] getLeastNumbers(int[] arr, int k) {if (k == 0 || arr.length == 0) {return new int[0];}// TreeMap的key是数字, value是该数字的个数。// cnt表示当前map总共存了多少个数字。TreeMap<Integer, Integer> map = new TreeMap<>();int cnt = 0;for (int num: arr) {// 1. 遍历数组,若当前map中的数字个数小于k,则map中当前数字对应个数+1if (cnt < k) {map.put(num, map.getOrDefault(num, 0) + 1);cnt++;continue;} // 2. 否则,取出map中最大的Key(即最大的数字), 判断当前数字与map中最大数字的大小关系://    若当前数字比map中最大的数字还大,就直接忽略;//    若当前数字比map中最大的数字小,则将当前数字加入map中,并将map中的最大数字的个数-1。Map.Entry<Integer, Integer> entry = map.lastEntry();if (entry.getKey() > num) {map.put(num, map.getOrDefault(num, 0) + 1);if (entry.getValue() == 1) {map.pollLastEntry();} else {map.put(entry.getKey(), entry.getValue() - 1);}}}// 最后返回map中的元素int[] res = new int[k];int idx = 0;for (Map.Entry<Integer, Integer> entry: map.entrySet()) {int freq = entry.getValue();while (freq-- > 0) {res[idx++] = entry.getKey();}}return res;}
}作者:sweetiee
链接:https://leetcode-cn.com/problems/zui-xiao-de-kge-shu-lcof/solution/3chong-jie-fa-miao-sha-topkkuai-pai-dui-er-cha-sou/

时间复杂度:O(NlogN) 空间复杂度:O(N)

【总结】

1.堆/快速选择比较

看起来分治法的快速选择算法的时间、空间复杂度都优于使用堆的方法,但是要注意到快速选择算法的几点局限性:

  • 算法需要修改原数组,如果原数组不能修改的话,还需要拷贝一份数组,空间复杂度就上去了。
  • 算法需要保存所有的数据。如果把数据看成输入流的话,使用堆的方法是来一个处理一个,不需要保存数据,只需要保存 k 个元素的最大堆。而快速选择的方法需要先保存下来所有的数据,再运行算法。当数据量非常大的时候,甚至内存都放不下的时候,就麻烦了。所以当数据量大的时候还是用基于堆的方法比较好
2. 一题多解 按照自己的掌握情况选择 熟悉or了解

参考链接:https://leetcode-cn.com/problems/zui-xiao-de-kge-shu-lcof/solution/tu-jie-top-k-wen-ti-de-liang-chong-jie-fa-you-lie-/

参考链接:https://leetcode-cn.com/problems/zui-xiao-de-kge-shu-lcof/solution/3chong-jie-fa-miao-sha-topkkuai-pai-dui-er-cha-sou/
image.png

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/425985.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

表数据的增删改查

一、插入数据 第一种插入方式 语法&#xff1a; insert into 表名(字段1,字段2....) values(值1,值2....) #往指定的表中添加数据&#xff1a;insert into 表名(字段1,字段2....) values(值1,值2....) INSERT INTO java成绩表(姓名,班级,Java成绩) VALUES(王二麻子,dt55班,90.5…

java学习(161):同步代码块

public class SynCode implements Runnable{public void run(){synchronized (this){Thread currentThread.currentThread();//获取当前线程for(int i1;i<10;i){System.out.println( "当前执行代码块的名称为" current.getName());try {Thread.sleep( 1000 );}cat…

ubuntu eclipse java,Ubuntu快速安装eclipse

Ubuntu快速安装eclipse教程&#xff0c;供大家参考&#xff0c;具体内容如下1.先安装jdk&#xff0c;详情见链接2.下载eclipse.tar.gz安装包(官网下载)3.把下载好的安装包复制到/usr目录下先cd到tar.gz压缩包的目录下cd /home/k/下载sudo cp eclipse.tar.gz /usr/4.解压tar.gz文…

[如何做研究][如何写论文]

音频 沈向洋&#xff1a;有效的科研法则 科学上网&#xff1a;https://www.youtube.com/watch?vU6r3R87AKHI&featureyoutu.be 视频 文章 【经典重温】MIT人工智能实验室: 如何做研究&#xff1f; 机器学习研究者的养成指南&#xff0c;吴恩达建议这么读论文 周志华 […

java学习(162):同步对象锁

定义一个dog类 public class Dog {private String name;public String getName() {return name;}public void setName(String name) {this.name name;} }定义一个同步对象锁 //同步对象锁 public class SysObject implements Runnable {private Dog dog;public SysObject(){…

word java api,是否有可以创建丰富Word文档的Java API?

2007年,我的项目成功使用OpenOffice.org的Universal Network Objects(UNO)界面,以编程方式从Java Web应用程序(Struts / JSP框架)生成MS-Word兼容文档(* .doc)以及相应的PDF文档.OpenOffice UNO还允许您构建与MS-Office兼容的图表,电子表格,演示文稿等.我们能够动态构建复杂的W…

[Leedcode][JAVA][第56题][合并区间][数组][贪心算法]

【问题描述】56.合并区间 给出一个区间的集合&#xff0c;请合并所有重叠的区间。 示例 1: 输入: [[1,3],[2,6],[8,10],[15,18]] 输出: [[1,6],[8,10],[15,18]] 解释: 区间 [1,3] 和 [2,6] 重叠, 将它们合并为 [1,6].【解答思路】 1. 双指针 左边位置一定是确定&#xff0c;…

java读写注册表的两种方式,Preferences与jRegistry

打开注册表方式&#xff1a;电脑上的windows键加r键打开运行程序->输入指令regedit->随即系统便会提示你是否要运行&#xff0c;选择“是“就行了 由于java程序是“write once, run everywhere”&#xff0c;用java读写注册表&#xff0c;那程序的跨平台性就差了。java对…

java学习(163):同步方法锁

public class SynMethon implements Runnable{private double money1000000;public void run(){dbMoney();}//同步方法private synchronized void dbMoney(){Thread tThread.currentThread();for(int i1;i<10;i){if(t.getName().equals( "会计" )){moneyi*10;Sy…

php缓存读取api,php - 如何根据返回的etag使用curl缓存api响应? - SO中文参考 - www.soinside.com...

这是一个快速的问题&#xff0c;也可能会对其他人有所帮助。我有一个rest服务&#xff0c;它将为端点上的每个GET查询返回一个正确的ETAG头。现在&#xff0c;我还需要创建一个curl http客户端来查询这些终结点并利用etags。据我了解&#xff0c;在使用curl发出第一个请求后&am…

PBFT

摘要&#xff1a; PBFT是Practical Byzantine Fault Tolerance的缩写&#xff0c;即&#xff1a;实用拜占庭容错算法。该算法是Miguel Castro&#xff08;卡斯特罗&#xff09;和Barbara Liskov&#xff08;利斯科夫&#xff09;在1999年提出来的&#xff0c;解决了原始拜占庭…

Python爬虫爬取美剧网站

一直有爱看美剧的习惯&#xff0c;一方面锻炼一下英语听力&#xff0c;一方面打发一下时间。之前是能在视频网站上面在线看的&#xff0c;可是自从广电总局的限制令之后&#xff0c;进口的美剧英剧等貌似就不在像以前一样同步更新了。 但是&#xff0c;作为一个宅diao的我又怎甘…

php 存储html 内容,HTML 本地存储

HTML 本地存储HTML 本地存储&#xff1a;优于 cookies。什么是 HTML 本地存储&#xff1f; ( 推荐学习&#xff1a;html教程 )通过本地存储(Local Storage)&#xff0c;web 应用程序能够在用户浏览器中对数据进行本地的存储。在 HTML5 之前&#xff0c;应用程序数据只能存储在 …

[Leedcode][JAVA][第55题][跳跃游戏][贪心][动态规划]

【问题描述】 给定一个非负整数数组&#xff0c;你最初位于数组的第一个位置。数组中的每个元素代表你在该位置可以跳跃的最大长度。判断你是否能够到达最后一个位置。示例 1:输入: [2,3,1,1,4] 输出: true 解释: 我们可以先跳 1 步&#xff0c;从位置 0 到达 位置 1, 然后再从…

java学习(165):inetaddress和inetsocketaddress

import java.net.InetAddress; import java.net.InetSocketAddress; import java.net.UnknownHostException;//inetaddress public class test112 {public static void main(String[] args){try {InetAddress add0 InetAddress.getLocalHost();//获取本机ip地址实例System.out…

如何删除oracle重复数据库,教你删除Oracle数据库中重复没用的数据

二、完全删除重复记录对于表中两行记录完全一样的情况&#xff0c;可以用下面语句获取到去掉重复数据后的记录&#xff1a;select distinct * from 表名可以将查询的记录放到暂时表中&#xff0c;然后再将原来的表记录删除&#xff0c;最初将暂时表的数据导回原来的表中。如下&…

java学习(166):socket服务端和客户端连接

socket服务端 import java.io.IOException; import java.net.ServerSocket; import java.net.Socket;//socket服务端 public class test114 {public static void main(String[] args){ServerSocket serverSocketnull;Socket clientnull;/*创建服务器套接字*/try {serverSocket…

[Leedcode][JAVA][第11题][盛最多水的容器][双指针][贪心]

【问题描述】11.盛最多水的容器 给你 n 个非负整数 a1&#xff0c;a2&#xff0c;...&#xff0c;an&#xff0c;每个数代表坐标中的一个点 (i, ai) 。在坐标内画 n 条垂直线&#xff0c;垂直线 i 的两个端点分别为 (i, ai) 和 (i, 0)。找出其中的两条线&#xff0c;使得它们与…

SAP S/4HANA使用ABAP获得生产订单的状态

在S/4HANA里&#xff0c;我们如何根据一个销售订单的行项目&#xff0c;查看对应的生产订单状态&#xff1f; 双击行项目&#xff1a; 点击Schedule line&#xff1a; 这里就能看到生产订单的ID和状态了。 其中订单的状态存储在表vsaufk里&#xff0c;注意订单和状态可以是1对多…

oracle 外部表 时间戳,Hive建立外部表与时间戳转换(含建dual表,修改列名,row_number() 函数等)...

建外部表&#xff0c;之前hadoop职位统计就是这么做的hive> drop table job;OKTime taken: 5.446 secondshive> show tables;OKTime taken: 0.018 secondshive> create external table job(area string, experience string, degree string,> num string, salary st…