【LeetCode笔记】215. 数组中的第K个最大元素(Java、快排、堆排、并发快排)

文章目录

  • 题目描述
  • 思路 & 代码
      • 快排
          • 基于 Fork / Join 的并发快排
          • 针对 topK 的快排优化
      • 堆排
          • 基本堆排
          • 结合题目的堆排
      • 二刷

题目描述

  • 大名鼎鼎的TOP K,主要考察排序
  • 快排 & 堆排
    在这里插入图片描述

思路 & 代码

快排

  • 没啥好说的,就是快排结束后,返回倒数第K个数字即可。
  • 重点就在于快排的实现了,好久没敲了= =:
  1. 原理:使用标志位进行分治的排序,每趟能让标志位的左边都不大于标志位,右边都不小于标志位。
  2. 注意点:边界问题、起始位置等,详见代码注释
  3. 留个坑:随机化mark,防止退化到 O(n2n^2n2)
class Solution {public int findKthLargest(int[] nums, int k) {myQuickSort(nums, 0, nums.length - 1);return nums[nums.length - k];}void myQuickSort(int[] nums, int left, int right){if(left >= right){return;}int mark = nums[left];int i = left, j = right;// 总共进行四次 i < j 的判断!while(i < j){// 从右边开始找,避免标志位本就应在最左边的情况导致出错。while(i < j && nums[j] >= mark){j--;}// 注意 ‘=’ 噢~while(i < j && nums[i] <= mark){i++;}if(i < j){int temp = nums[i];nums[i] = nums[j];nums[j] = temp;}}// 结束后 i == j,此时nums[i] 一定不大于 nums[left](可证)// 把标志位放到正确的位置nums[left] = nums[i];nums[i] = mark;myQuickSort(nums, left, i - 1);myQuickSort(nums, i + 1, right);}
}
基于 Fork / Join 的并发快排
  • 效率直接从 34s 提高到 8s,太狠了
  • fork / join简直是快排的指定好兄弟,又好用效率又高。
class Solution {public int findKthLargest(int[] nums, int k) {// 开池,放任务ForkJoinPool forkJoinPool = new ForkJoinPool(Runtime.getRuntime().availableProcessors());forkJoinPool.invoke(new SortTask(nums, 0, nums.length - 1));return nums[nums.length - k];}class SortTask extends RecursiveAction {int[] arr;int left;int right;public SortTask(int[] arr, int left, int right) {this.arr = arr;this.left = left;this.right = right;}@Overrideprotected void compute() {// 1、任务结束if(left >= right) return;// 2、当前任务执行int mark = arr[left];int i = left, j = right;while(i < j) {while(i < j && arr[j] >= mark) j--;while(i < j && arr[i] <= mark) i++;if(i < j) {int temp = arr[i];arr[i] = arr[j];arr[j] = temp;}}arr[left] = arr[i];arr[i] = mark;// 3、任务拆分、执行SortTask leftTask = new SortTask(arr, left, i - 1);SortTask rightTask = new SortTask(arr, i + 1, right);leftTask.fork();rightTask.fork();leftTask.join();rightTask.join();}}
}
针对 topK 的快排优化
  • 能达到 Fork / Join 的效率,并且不需要额外线程资源
  • 关键在于局部有序,对于 topK 问题来说,我们只关心第K个最大元素。
  • 因此实际上并不需要做到全局有序,可以在每次递归时都只选择一个区间进行递归
  • 时间复杂度:等待有缘人补充
class Solution {public int findKthLargest(int[] nums, int k) {sort(nums, 0, nums.length - 1, k);return nums[nums.length - k];}void sort(int[] nums, int left, int right, int k) {if(left >= right) return;int mark = nums[left];int i = left, j = right;while(i < j) {while(i < j && nums[j] >= mark) j--;while(i < j && nums[i] <= mark) i++;if(i < j) {int temp = nums[i];nums[i] = nums[j];nums[j] = temp;}}nums[left] = nums[i];nums[i] = mark;int rightNums = right - i + 1;if(rightNums < k) sort(nums, left, i - 1, k - rightNums); // 走左,右边全部拿走else sort(nums, i + 1, right, k); // 走右,不变}
}

堆排

  • 可以直接用堆排,也可以针对题目变形一下(会快很多)。这里都贴一下
基本堆排
  • 三个主要函数:heapify、buildTree 与 myHeapSort
  • 逻辑上是完全二叉树,通过数组实现
  • Parent = (son - 1) / 2
  • Son1 = Parent * 2 + 1。 Son2 = Son1 + 1。
  • heapify:递归向下
  • buildTree:从下(第一个Parent)往上遍历进行heapify
  • myHeapSort:每次都取最值,与结尾交换。然后缩小范围继续。
class Solution {public int findKthLargest(int[] nums, int k) {// 堆排序,类似快排做法myHeapSort(nums);return nums[nums.length - k];}void swap(int[] nums, int one, int two){int temp = nums[one];nums[one] = nums[two];nums[two] = temp;}// 自顶向下;n 不一定等于 nums.length 噢~void heapify(int[] nums, int n, int now){// 递归结束条件if(now >= n){return;}// 选出最大值对应下标:两次对比int maxIndex = now;int son1 = 2 * now + 1;int son2 = son1 + 1;// 判断是否超出边界。if(son1 < n && nums[maxIndex] < nums[son1]){maxIndex = son1;}if(son2 < n && nums[maxIndex] < nums[son2]){maxIndex = son2;}// 需要继续的情况if(maxIndex != now){swap(nums, now, maxIndex);heapify(nums, n, maxIndex);}}// 自底向上void buildTree(int[] nums){int lastNode = nums.length - 1;int lastParent = (lastNode - 1) / 2;// 自下而上,自右而左进行 heapifyfor(; lastNode >= 0; lastNode--){heapify(nums, nums.length, lastNode);}}void myHeapSort(int[] nums){buildTree(nums);int lastNode = nums.length - 1;for(int i = lastNode; i >= 0; i--){swap(nums, 0, i);heapify(nums, i, 0);}}
}
结合题目的堆排
  • 找到第K大元素的时候直接返回值。
class Solution {public int findKthLargest(int[] nums, int k) {// 堆排序,结合题目的改进return myHeapSort(nums, k);}void swap(int[] nums, int one, int two){int temp = nums[one];nums[one] = nums[two];nums[two] = temp;}void heapify(int[] nums, int n, int now){// 递归结束条件if(now >= n){return;}// 选出最大值对应下标:两次对比int maxIndex = now;int son1 = 2 * now + 1;int son2 = son1 + 1;// 判断是否超出边界。if(son1 < n && nums[maxIndex] < nums[son1]){maxIndex = son1;}if(son2 < n && nums[maxIndex] < nums[son2]){maxIndex = son2;}// 需要继续的情况if(maxIndex != now){swap(nums, now, maxIndex);heapify(nums, n, maxIndex);}}void buildTree(int[] nums){int lastNode = nums.length - 1;int lastParent = (lastNode - 1) / 2;// 自下而上,自右而左进行 heapifyfor(; lastNode >= 0; lastNode--){heapify(nums, nums.length, lastNode);}}// 排到第k个的时候直接结束。int myHeapSort(int[] nums, int k){buildTree(nums);int lastNode = nums.length - 1;for(int i = lastNode; i >= 0; i--, k--){if(k == 1){return nums[0];}swap(nums, 0, i);heapify(nums, i, 0);}return -1;}
}

二刷

  • 快排:右边开始、等于号,还是这两个注意点。。
while(i < j){// 从右边开始找,避免标志位本就应在最左边的情况导致出错。while(i < j && nums[j] >= mark) j--;// 没有 '=' 会造成死循环while(i < j && nums[i] <= mark) i++;
  • 堆排:三个主要函数。一个自顶向上,一个自底向上。
class Solution {public int findKthLargest(int[] nums, int k) {heapSort(nums);return nums[nums.length - k];}// heapSort:先建树,然后从后往前,逐个交换、维护堆。void heapSort(int[] nums) {buildTree(nums);for(int i = nums.length - 1; i >= 0; i--) {swap(nums, i, 0);heapify(nums, i, 0);}}// buildTree:自底向上进行建树void buildTree(int[] nums) {for(int son = nums.length - 1; son >= 0; son--) {heapify(nums, nums.length, son);}}// heapify:自顶向下进行堆维护,注意边界判断void heapify(int[] nums, int length, int nowIndex) {if(nowIndex >= length) {return;}int son1 = nowIndex * 2 + 1;int son2 = son1 + 1;int maxIndex = nowIndex;if(son1 < length && nums[son1] > nums[maxIndex]) {maxIndex = son1;}if(son2 < length && nums[son2] > nums[maxIndex]) {maxIndex = son2;}if(maxIndex != nowIndex) {swap(nums, maxIndex, nowIndex);heapify(nums, length, maxIndex);}}void swap(int[] nums, int left, int right) {int temp = nums[left];nums[left] = nums[right];nums[right] = temp;}
}

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

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

相关文章

c语言中怎么画直线,ps如何画直线 【操作流程】

大家都知道电脑使用起来非常方便&#xff0c;但遇见ps如何画直线的时候就非常头疼了&#xff0c;如果你是第一次遇到ps如何画直线&#xff0c;怎么样才能快速解决ps如何画直线带来的烦恼呢&#xff1f;小编为大家收集了很多关于ps如何画直线问题的解决方法&#xff0c;下面请看…

【LeetCode笔记】470. 用Rand7()实现Rand10()(Java、概率)

文章目录题目描述代码 & 思路更新版三刷 - 极简版题目描述 貌似是比较高频的面试题目啊&#xff0c;有学概率论内味了&#xff08;讲道理我概率论学得不好&#xff09; 代码 & 思路 先用Rand7实现RandN&#xff08;N > 10&#xff09;&#xff0c;类似进制&…

【纠错记录】本地FTP服务器无法被外部连接

问题描述 在本地主机开了个FTP服务器&#xff0c;本机可以正常访问&#xff0c;但是外部主机不能访问FTP服务器 解决方法 ① 一开始以为是服务端Xlight FTP的问题&#xff0c;检查权限并没有发现问题 ② 接着考虑到有没有可能是客户端Flash FXP的问题&#xff0c;于是尝试直…

双线性插值算法c 语言实现,双线性插值 - CristianoC的个人空间 - OSCHINA - 中文开源技术交流社区...

前言最近在学习LBP&#xff0c;其中的圆形LBP特征由于计算的值可能不是整数&#xff0c;即计算出来的点不在图像上&#xff0c;我们要使用计算出来的点的插值点。目的的插值方法有很多&#xff0c;Opencv使用的是双线性插值&#xff0c;今天就来介绍一下双线性插值。概述先给大…

android 访问服务器josn文件,Android客户端对服务器回来的json文件进行解析

Android客户端对服务器返回的json文件进行解析和解析XML的方式大同小异&#xff0c;只有解析方式存在区别:/*** 解析服务器返回来的json数据* param content* return* throws Exception*/private static List parseJson(String content) throws Exception {List videoList new…

【LeetCode笔记】543. 二叉树的直径(Java、dfs、二叉树)

文章目录题目描述思路 & 代码题目描述 思路 & 代码 由这个结论考虑&#xff1a;直径中一定有一个父结点&#xff0c;那么当前直径长度就是&#xff1a; 当前父结点的左子树深度 右子树深度那么&#xff0c;只要遍历所有结点的最长直径值即可流程&#xff1a;在找每一…

【LeetCode笔记】461. 汉明距离(Java、位运算)

文章目录题目描述思路 & 代码更新版题目描述 既然是二进制&#xff0c;那就十有八九用位运算符既然是不同&#xff0c;那肯定用异或 思路 & 代码 两种方法&#xff0c;总体来说都是处理异或值方法一&#xff0c;转换成字符串&#xff0c;再遍历&#xff08;会慢一点…

android 启动服务同时传递数据,Android Studio开发基础之起动Service,并通过从Activity向Service传递数据...

Android Studio开发基础之启动Service&#xff0c;并通过从Activity向Service传递数据本实例演示启动Service&#xff0c;并通过从Activity向Service传递数据&#xff0c;新建一个Service&#xff0c;并敲如下代码&#xff1a;package com.example.lhb.startservice;import and…

android 自定义桌面图标大小设置,手机桌面图标尺寸可以改?OriginOS重新定义个性化...

原标题&#xff1a;手机桌面图标尺寸可以改&#xff1f;OriginOS重新定义个性化随着安卓定制化系统同质化情况的加重&#xff0c;越来越多的终端厂商开始探索一场新的变革。深耕手机领域的vivo&#xff0c;自然也察觉到了这一点。经过不断地努力和尝试之后&#xff0c;vivo终于…

【LeetCode笔记】347. 前K个高频元素(Java、优先队列、小顶堆、HashMap)

文章目录题目描述思路 & 代码更新版&#xff1a;引入 stream 流 Lambda题目描述 时间复杂度小于O(n*logn)&#xff0c;否则直接sort&#xff0c;再遍历就很轻松。很有学习价值的题目&#xff0c;第一次使用了优先队列PriorityQueue。 思路 & 代码 首先遍历数组&am…

signature=c0b9be9cdeb0a9e14dbbc8edc1d4c0e5,NRZ-PM-QPSK 16

摘要&#xff1a;We present a set of experiments of 16 100 Gb/s (1.6 Tb/s) coherent wavelength-division-multiplexing nonreturn-to-zero polarization-multiplexed quadrature phase-shift-keying transmission over installed standard single-mode fiber, probing dif…

CentOS 7.9 vmware workstation安装

https://mirrors.aliyun.com/centos/7.9.2009/isos/x86_64/CentOS-7-x86_64-Everything-2009.iso VMWare workstation 可以自动化安装这个版本的Linux&#xff0c;节省我很多时间。

【LeetCode笔记】240. 搜索二维矩阵II 剑指 Offer 04 二维数组中的查找(Java、指针)

文章目录题目描述思路 & 代码更新版题目描述 两道题一样的有序 搜索&#xff0c;明摆着二分查找但是比较懒…选取了题解里的指针做法 思路 & 代码 时间复杂度O(m n)&#xff0c;原理 & 思路 & 证明可见代码注释。代码很简单&#xff0c;主要是弄明白正确…

html中如何写平方根等,平方根的符号怎么打出来 电脑打字,平方根的符号怎样打出来?...

在WORD里面如何输入平方根“符号”&#xff1f;注&#xff1a;不是要公式&#xff0c;是要一个符号&#xff0c;最好回答时能打出来。方便下复制&#xff0c;哈哈用word域来完成&#xff0c;按下热键ctrlF9&#xff0c;然后输入 EQ \R(b,c)&#xff0c; 再按F9切换域代码即可&a…

【LeetCode笔记】238. 除自身以外数组的乘积(Java、思路题)

文章目录题目描述思路 & 代码更新版题目描述 首先暴力二层循环肯定可以&#xff0c;然后先累乘整个数组&#xff0c;再用除法肯定也行。但是很遗憾&#xff0c;两种做法都不满足题目说明。 思路 & 代码 O(n) & O(1)考虑到这个核心&#xff1a;ans[i] 就是当前元…

js在div中追加html,JS把内容动态插入到DIV的实现方法

/p>"http://www.w3.org/TR/xhtml11/DTD/xhtml11.dtd">Testingexample.js 文件内容&#xff1a;window.onload function() {var testdiv document.getElementById("testdiv");testdiv.innerHTML"I inserted this content.";}另一段代码&a…

【LeetCode笔记】236. 二叉树的最近公共祖先(Java、二叉树、DFS)

文章目录题目描述思路 & 代码题目描述 ac了&#xff0c;但是看了题解发现有更好的做法 思路 & 代码 对于一个结点rootNode&#xff0c;p、q有这么几种情况&#xff1a; p、q分别在其左右子树中&#xff0c;那么好说&#xff0c;rootNode就是答案p、q都在其右子树中…

产生复选框的html,html根据复选框勾选顺序生成字符串

check猴子大象狮子String.prototype.trim function (c) {if (!c) c "\\s";return this.replace(new RegExp("(^" c ")|(" c "$)", "ig"), "");};function chkTag(e) {var val "";if (e.checked)…

linux关机命令_Linux基于centOS 7.6常见的Linux命令

一、linux关机命令&#xff1a;1.shutdown命令安全地将系统关机&#xff08;推荐&#xff09;参数说明:[-r] 重启计算器。[-h] 关机后关闭电源〔halt〕。[-c] cancel current process取消目前正在执行的关机程序。[-time] 设定关机〔shutdown〕前的时间。shutdown -h now 立刻…

【LeetCode笔记】221. 最大正方形(Java、动态规划、思路题)

文章目录题目描述思路 & 代码更新版题目描述 显而易见地可以用dp来写&#xff0c;问题在于如何考虑状态转移方程 思路 & 代码 首先再加一层外墙&#xff0c;就不用边界判断了maxSqure[i]&#xff1a;以当前格子为右下角的正方形&#xff0c;可达到的最大边长这是由…