剑指offer——最小的K个数和数组中第K大的元素

 

 

解题思路:

乘着做这个题,顺便复习下堆排序。

先说堆排序是一个什么东西:https://blog.csdn.net/u013384984/article/details/79496052

大顶堆升序,小顶堆降序。

随便定义的一个堆。

 

第一步:

此时我们从最后一个非叶子结点开始(叶结点自然不用调整,第一个非叶子结点 arr.length/2-1=5/2-1=1,也就是下面的6结点),从左至右,从下至上进行调整。

此处必须注意,我们把6和9比较交换之后,必须考量9这个节点对于其子节点会不会产生任何影响?因为其是叶子节点,所以不加考虑;但是,一定要熟练这种思维,写代码的时候就比较容易理解为什么会出现一次非常重要的交换了。

 

 

 注意:第一步已经把9弄上去了,所以只需要看9 5 6这三个数字是不是符合大顶堆,一看符合的,然后回到4.找到第二个非叶节点4,由于[4,9,8]中9元素最大,4和9交换。

牢记上面说的规则,每次交换都要把改变了的那个节点所在的树重新判定一下,这里就用上了,4和9交换了,变动了的那棵子树就必须重新调整,一直调整到符合大根堆的规则为截。

此时,我们就将一个无序序列构造成了一个大顶堆。

步骤二 将堆顶元素与末尾元素进行交换,使末尾元素最大。然后继续调整堆,再将堆顶元素与末尾元素交换,得到第二大元素。如此反复进行交换、重建、交换。

a.将堆顶元素9和末尾元素4进行交换

这里,必须说明一下,所谓的交换,实际上就是把最大值从树里面拿掉了,剩下参与到排序的树,其实只有总结点的个数减去拿掉的节点个数了。所以图中用的是虚线。

 

注意:这里和root交换后的东西,后面已经不放在堆里面进行排了。

 1 import java.util.ArrayList;
 2 public class Solution {
 3     public ArrayList<Integer> GetLeastNumbers_Solution(int [] input, int k) {
 4         
 5         ArrayList<Integer> res = new ArrayList<>();
 6         
 7         if(k>input.length || input==null)
 8             return res;
 9         // 按照完全二叉树的特点,从最后一个非叶子节点开始,对于整棵树进行大根堆的调整
10         // 也就是说,是按照自下而上,每一层都是自右向左来进行调整的
11         // 注意,这里元素的索引是从0开始的
12         // 另一件需要注意的事情,这里的建堆,是用堆调整的方式来做的
13         // 堆调整的逻辑在建堆和后续排序过程中复用的
14         for(int i=k/2-1;i>=0;i--)
15         {
16             Adjust(input,i,k-1);//从最后一个非叶子节点开始
17         }
18         
19         for(int j=k;j<input.length;j++)
20         {
21             // 元素交换之后,毫无疑问,最后一个元素无需再考虑排序问题了。
22             // 接下来我们需要排序的,就是已经去掉了部分元素的堆了,这也是为什么此方法放在循环里的原因
23             // 而这里,实质上是自上而下,自左向右进行调整的
24             if(input[j]<input[0])
25             {
26                 int temp = input[0];
27                 
28                 input[0] = input[j];
29                 
30                 input[j] = temp;
31                 
32                 Adjust(input,0,k-1);
33             }
34         }
35         
36         for(int i=0;i<k;i++)
37         {
38             res.add(input[i]);
39         }
40         return res;
41     }
42     public void Adjust(int [] input,int k,int length)
43     {
44         int temp = input[k];
45         
46         for(int i=2*k+1;i<=length;i=2*i+1)//i是子节点
47         {
48             if(i<length && input[i]<input[i+1])
49                 i++;
50             
51             if(temp>input[i])
52             {
53                 break;
54             }
55             else
56             {
57                 swap(input, i, k);
58                 // 下面就是非常关键的一步了
59                 // 如果子节点更换了,那么,以子节点为根的子树会不会受到影响呢?
60                 // 所以,循环对子节点所在的树继续进行判断
61                 k = i;
62             }
63         }
64     }
65     
66     public void swap(int []input,int a,int b)
67     {
68         int temp = input[a];
69         input[a]=input[b];
70         input[b] = temp;
71     }
72     
73     
74 }

 

 1 class Solution {
 2     public int findKthLargest(int[] input, int k) {
 3 
 4         if(input.length==1)
 5             return input[0];
 6         int length = input.length;
 7         for(int i=k/2-1;i>=0;i--)
 8         {
 9             Adjust(input,i,length-1);
10         }
11         
12         for(int j = length - 1; j >=k; j--)
13         {
14             if(input[j]>input[0])
15             {
16                 int temp = input[0];
17                 input[0] = input[j];
18                 input[j] = temp;
19                 Adjust(input,0,k-1);
20             }
21         }
22 
23         return input[0];
24         
25             
26     }
27     
28     public void Adjust(int []input,int k,int length)
29     {
30         int temp = input[k];
31         for(int i=2*k+1;i<=length;i=2*i+1)
32         {
33             if(i<length && input[i]>input[i+1])
34             {
35                 i++;
36             }
37             if(temp<input[i])
38             {
39                 break;
40             }
41             else
42             {
43                 swap(input,i,k);
44                 k=i;
45             }
46             
47         }
48     }
49     
50     public void swap(int []input,int a,int b)
51     {
52         int temp = input[a];
53         input[a]=input[b];
54         input[b] = temp;
55     }
56     
57 }

 

转载于:https://www.cnblogs.com/wangyufeiaichiyu/p/11054807.html

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

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

相关文章

深度启动盘工具linux,Linux下无法安装深度启动盘制作工具

正在选中未选择的软件包 deepin-boot-maker。(正在读取数据库 ... 系统当前共安装有 231455 个文件和目录。)正准备解包 deep.deb ...正在解包 deepin-boot-maker (2.0.0-1) ...dpkg: 依赖关系问题使得 deepin-boot-maker 的配置工作不能继续&#xff1a;deepin-boot-maker 依赖…

GIS二次开发之初探

最近一段时间在学习GIS的二次开发&#xff0c;作为新手&#xff0c;最好的方法就是泡论坛&#xff0c;看原码&#xff0c;当然涉及到地理方面的专业知识&#xff0c;不可能说一两天就可以弄懂&#xff0c;只有在学习编程的过程中慢慢的去体会一些地理专业上面的知识。 就几天的…

构建之法阅读笔记03

个人感受&#xff1a; 过去&#xff1a;个人英雄主义&#xff0c;却是一只狗熊 为什么不好&#xff1a;双拳难敌四腿&#xff0c;大项目一个人是很难完成的 解决方法&#xff1a;加入团队&#xff0c;共同开发 这章让我了解到了很多种软件开发的团队模式&#xff0c;有娱乐性的…

CSS3 2D Transform

位移translate()函数 结合起来&#xff0c;translate()函数移动元素主要有以下三种移动&#xff1a; 水平移动&#xff1a;向右移动translate(tx,0)和向左移动translate(-tx,0)&#xff1b;垂直移动&#xff1a;向上移动translate(0,-ty)和向下移动translate(0,ty);对角移动&am…

[转]用g++编译动态链接库

写了一个最简单的动态链接库程序&#xff0c;使用g命令行编译。怕以后忘记&#xff0c;就把它记到blog中。动态库导出头文件:/** * file: dll.h * Powered by JGood 2009-09-22 */ #ifndef __dll_h__#define __dll_h__ #ifdef __MY_DLL_LIB__ #define DLL_EXPORT extern …

linux传奇源码,游戏源码 屠龙传奇H5 鲲乃异兽 全套源码+教程

一、游戏的要求系统要求系统要求liunx6.9系统 因为游戏是java对服务器配置要求比较高如果您的资金够宽裕可以选择直接购买 liunx6.9系统 8核16G 240G/320G硬盘 10兆以上宽带的服务器 这样可以直接保证游戏的稳定性如果不想出那么多钱 可以一步一步来开新区用4核4G 1…

javascript setTimeout 和 setInterval 区别

[setTimeout]setTimeout(表达式,延时时间)在执行时,是在载入后延迟指定时间后,去执行一次表达式,记住,次数 是一次用setTimeout实现的自动变化显示随机数的效果:<html><head><script>window.οnlοadsett;function sett(){document.body.innerHTMLMath.rando…

如何量化考核软件开发人员绩效

为什么80%的码农都做不了架构师&#xff1f;>>> 1、首先有技术积累的前提下&#xff0c;能比较正确的估量项目成本和项目组生产率 2、设立项目组短期目标、中期、长期目标&#xff08;这个因项目具体情况而已&#xff0c;有些项目都是很小的&#xff0c;甚至2个星期…

linux创建自定义组件qt,关于QT自定义控件

Linux中I/O设备分为两类&#xff1a;块设备和字符设备。两种设备本身没有严格限制&#xff0c;但是&#xff0c;基于不同的功能进行了分类。用户自定义的控件可以通过继承现有的 Qt 控件实现&#xff0c;也可以直接从 QWidget 继承。QT中的label控件&#xff0c;没有预定义的单…

jQuery三级下拉菜单

演示地址:http://www.corange.cn/demo/3738/index.html <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"> <html xmlns"http://www.w3.org/1999/xhtml"&g…

100个高质量Java开发者博客

ImportNew注&#xff1a;原文中还没有100个。作者希望大家一起来推荐高质量的Java开发博客&#xff0c;然后不段补充到这个列表。欢迎你也参与推荐优质的Java开发博客。&#xff08;声明一下&#xff1a;我们的数学不是体育老师教的&#xff01;:) &#xff09; 本文的主要目的…

IT外包概要

IT外包 前两天和朋友聊起这个外包的问题&#xff0c;就顺便给他说了一下&#xff0c;自己也整理了一下&#xff0c;发出来&#xff0c;方便更多的人。 如果有说的不准确的地方欢迎大家补充分享。 大致分两种&#xff1a; 项目外包&#xff0c;人力外包。简而言之&#xff1a;项…

linux gcc march arch,Gcc的spec中arch什么的指定

gcc/gcc.c中的set_collect_gcc_options这里面有COLLECT_GCC_OPTIONS-marcharmv4t -mtunearm920t(gdb) p switches[0]$15 {part1 0x80792b1 "marcharmv4t", args 0x0, live_cond 1,validated 1 \001, ordering 0 \000}(gdb) p switches[1]$16 {part1 0x80792c…

php如何解决中文乱码问题?

为什么会出现中文乱码&#xff1f; 很多新手朋友学习PHP的时候&#xff0c;发现程序中的中文在输出的时候会出现乱码的问题&#xff0c;那么为什么会出现这种乱码的情况呢&#xff1f;一般来说&#xff0c;乱码的出现有2种原因&#xff0c;一种是由于编码(charset) 设置错误&am…

红帽将召开“开放你的世界”在线论坛

国外媒体报道&#xff0c;美国时间5月27日&#xff0c;红帽公司将召开在线开源论坛&#xff0c;主题为“开放你的世界”。时间为美国东部时间8:45-17:30.该论坛讨论的议题包括opensource.com网站所覆盖的健康医疗、教 育、政府、法律等领域。论坛嘉宾包括这些领域的开源思考者及…

sed命令的使用和说明

sed的三种用法1.shell命令行输入命令 格式&#xff1a;sed [选项] ’sed命令‘ 输入文件2.将sed命令出入脚本文件后&#xff0c;sed命令调用格式&#xff1a; sed [选项] -f sed脚本文件 输入文件3.将sed命令插入脚本文件后&#xff0c;三种脚本文件可执行&#xff0c;直接执行…

linux pam模块 cron,Linux-PAM 1.1.2 中文文档 - 6.32. pam_tally-登录计数器(统计)模块 | Docs4dev...

pam_tally.so [file * 2 *] [onerr [* 3 * | * 4 *]] [magic_root] [even_deny_root_account] [deny * 5 *] [lock_time * 6 *] [unlock_time * 7 *] [per_user] [no_lock_time] [no_reset] [审核] [静音] [no_log_info]pam_tally [--file * 9 *] [--user * 10 *] [--rese…

Java提取文本文档中的所有网址(小案例介绍正则基础知识)

正则表达式基础以及Java中使用正则查找 定义&#xff1a; 正则表达式是一些用来匹配和处理文本的字符串 正则的基础&#xff08;先大致了解下&#xff09; 1. 正则表达式的作用 查找特定的信息&#xff08;搜索&#xff09;替换一些文本&#xff08;替换&#xff09;2. 正则基础…

计算机鼓轮

计算机鼓轮是一种计算机早期应用的机械式模数转换器转载于:https://www.cnblogs.com/emanlee/archive/2010/12/08/1900053.html

字符串大小写字母转换c 语言,towlower()

wint_t towlower ( wint_t wc );towlower() 函数用来将大写字母转换为小写字母(针对宽字符)。只有当参数 wc 是一个大写字母&#xff0c;并且存在对应的小写字母时&#xff0c;这种转换才会发生。towlower() 是 tolower() 的宽字符版本。参数wc要被转换的宽字符。它可以是一个有…