快速排序题目SelectK问题(力扣75.颜色分类、力扣215.数组中的第K个最大元素、面试题17.14最小K个数)

力扣75.颜色分类

给定一个包含红色、白色和蓝色、共 n 个元素的数组 nums ,原地对它们进行排序,使得相同颜色的元素相邻,并按照红色、白色、蓝色顺序排列。

我们使用整数 0、 1 和 2 分别表示红色、白色和蓝色。

必须在不使用库内置的 sort 函数的情况下解决这个问题。

class Solution { //时间复杂度为O(n)//其实,变量 zero 相当于我们在三路快速排序算法中的 lt;//变量 two 相当于我们在三路快速排序算法中的 gt。public void sortColors(int[] nums) {// nums[0...zero] == 0, nums[zero + 1, i] == 1, nums[two, n - 1] == 2// 定义三个指针:zero、i、two,分别表示0的最右边界、当前处理的元素、2的最左边界int zero = -1, i = 0, two = nums.length;while(i < two){// 当前处理元素的指针小于2的最左边界时,继续循环// 如果当前元素为0,将其与zero右边的元素交换,并将zero和i都向右移动一位if(nums[i] == 0){zero++;swap(nums,zero,i);i++;}// 如果当前元素为2,将其与two左边的元素交换,并将two向左移动一位else if (nums[i] == 2){ // 注意此时不需要i右移,因为交换后的元素还需要继续判断two --;swap(nums, i, two);}else{ //如果当前元素是1,不需要操作,直接继续向右遍历i ++;}}}// 交换数组中指定位置的两个元素private void swap(int[] nums, int i, int j){int t = nums[i];nums[i]= nums[j];nums[j] = t;}
}

关于SelectK问题:即在一个无序数组中,找出第K小(大)的元素

我们首先来封装一个 selectK 的方法。封装好了这个方法以后,这三个问题都可
以快速求解。
我们的 selectK 的接口是这样的:

// 在 arr[l...r] 的范围里求解整个数组的第 k 小元素并返回
// k 是索引,即从 0 开始计算
int selectK(int[] arr, int l, int r, int k, Random rnd)

因为我们的 partition 过程需要随机选取标定点,所以我们还需要传一个 Random(快排的优化)
类的对象 rnd。
定义好函数签名以后,下面我们来书写相应的逻辑。

  • 首先,selectK 的过程,我们就是要执行一遍 partition。在这里,我使用双路快速排序的 partition。(partition属于核心代码块要掌握)
  • 注意,因为在这个问题中,我们肯定我们处理的数据类型是 int,所以,在代码中,我不再使用泛型:
private int partition(int[] arr, int l, int r, Random rnd){// 生成 [l, r] 之间的随机索引int p = l + rnd.nextInt(r - l + 1);swap(arr, l, p);// arr[l+1...i-1] <= v; arr[j+1...r] >= vint i = l + 1, j = r;while(true){while(i <= j && arr[i] < arr[l])i ++;while(j >= i && arr[j] > arr[l])j --;if(i >= j) break;swap(arr, i, j);i ++;j --;
}swap(arr, l, j);return j;
}
private void swap(int[] arr, int i, int j){int t = arr[i];arr[i] = arr[j];arr[j] = t;
}

有了 partition,我们的 selectK 的主题逻辑非常简单。

首先,进行 partition,假设结果是 p。我们只需要将 k 和 p 做比较。

  • 如果 k == p,直接返回 arr[p] 即可;
  • 如果 k < p,在 arr[l, p - 1] 的范围继续找,即调用 selectK(arr, l, p - 1, k, rnd);
  • 如果 k > p,在 arr[p + 1, r] 的范围继续找,即调用 selectK(arr, p + 1, r, k, rnd);

就有了下面的代码:

private int selectK(int[] arr, int l, int r, int k, Random rnd){
int p = partition(arr, l, r, rnd);
if(k == p) return arr[p];
if(k < p) return selectK(arr, l, p - 1, k, rnd);return selectK(arr, p + 1, r, k, rnd);
}

这样,我们就完成了 select K 的过程。是不是非常简单!

下面,我们用我们写的 select K,先来解决 Leetcode 上第 215 号问题:

这个问题是求第 k 大元素,但是我们的 selectK 求得是第 k 小元素。怎么办?
非常简单,我们只需要在调用 select K 之前,将求第 k 大元素的这个 k,转换成
对应求的是第几小元素对应的索引就好了。
按照题目描述,如果 k 是 1,对应就是要找最大元素,那么相应的我们的
select K 的索引,就是 nums.length - 1。(如果10个数,K=1,第一个最大的数,就是SelectK索引为9的那个的元素)
如果 k 是 nums.length,其实就是求最小元素,那么相应的我们的 selectK 的
索引,就是 0。  (如果10个数,第10个最大的数,就是SelectK索引为0的那个的元素,最小值)
他们之间的转换关系是 nums.length - k。

力扣215.数组中的第K个最大元素 

给定整数数组 nums 和整数 k,请返回数组中第 k 个最大的元素。

请注意,你需要找的是数组排序后的第 k 个最大的元素,而不是第 k 个不同的元素。

你必须设计并实现时间复杂度为 O(n) 的算法解决此问题。

import java.util.Random; //导入Random包
class Solution {public int findKthLargest(int[] nums, int k) {//只有两行,其他的内容全部复用我们上面实现的 selectK,是不是很酷?//我们的SelectK是第K最小元素,所以这里findKthLargest传入下标要处理一下//转换关系是 nums.length - kRandom rnd = new Random();return selectK(nums, 0, nums.length - 1, nums.length - k, rnd);}//有了 partition,我们的 selectK 的主题逻辑非常简单。private int selectK(int[] arr, int l, int r, int k, Random rnd){//首先,进行 partition,假设返回值结果是 p。我们只需要将 k 和 p 做比较。int p = partition(arr, l, r, rnd); //如果 k == p,直接返回 arr[p] 即可;if(k == p) return arr[p]; //如果 k < p,在 arr[l, p - 1] 的范围继续找,即调用 selectK(arr, l, p - 1, k, rnd);if(k < p) return selectK(arr, l, p - 1, k, rnd); //如果 k > p,在 arr[p + 1, r] 的范围继续找,即调用 selectK(arr, p + 1, r, k, rnd);return selectK(arr, p + 1, r, k, rnd);
}private int partition(int[] arr, int l, int r, Random rnd){// 生成 [l, r] 之间的随机索引int p = l + rnd.nextInt(r - l + 1);swap(arr, l, p);// arr[l+1...i-1] <= v; arr[j+1...r] >= vint i = l + 1, j = r;while(true){while(i <= j && arr[i] < arr[l])i ++;while(j >= i && arr[j] > arr[l])j --;if(i >= j) break;swap(arr, i, j);i ++;j --;}swap(arr, l, j);return j;}//数组指定索引,两数交换private void swap(int[] arr, int i, int j){int t = arr[i];arr[i] = arr[j];arr[j] = t;}
}

面试题17.14最小K个数&&剑指offer40&&LCR159.库存管理

 

下面,我们解决《剑指 Offer》上的 40 号问题,最小的 k 个数: 
https://leetcode-cn.com/problems/zui-xiao-de-kge-shu-lcof/ 
对于这个问题,我们只要使用上面的 selectK,找到第 k 小的数。
然后,selectK 的过程由于调用了 partiton,所以会调整整个数组的内容。
此时,这个第 k 小的 数的前面所有元素,就是整个数组最小的 k 个数字了。
 

这里要注意,我们求的第 k 小的数字,对应的索引是 k - 1,因为题目给的下标k从0开始

所以我们调用 selectK 的时候,需要传入的索引参数是 k - 1。 
另外,对于这个问题,k 可能取 0,此时,我们不需要执行 selectK,直接返回一 
个含有 0 个元素的new int[] 就好了。

怎么样,是不是很简单?具体过程如下:

import java.util.Arrays;
import java.util.Random;
class Solution {public int[] getLeastNumbers(int[] arr, int k) {if(k == 0) return new int[0];Random rnd = new Random();selectK(arr, 0, arr.length - 1, k - 1, rnd);
//Arrays.copyOf()是java.util.Arrays 类中的一个方法。它会创建一个指定长度的新数组,并将原始数组中的元素复制到新数组中。return Arrays.copyOf(arr, k); }private int selectK(int[] arr, int l, int r, int k, Random rnd){int p = partition(arr, l, r, rnd);if(k == p) return arr[p];if(k < p) return selectK(arr, l, p - 1, k, rnd);return selectK(arr, p + 1, r, k, rnd);}private int partition(int[] arr, int l, int r, Random rnd){// 生成 [l, r] 之间的随机索引int p = l + rnd.nextInt(r - l + 1);swap(arr, l, p);// arr[l+1...i-1] <= v; arr[j+1...r] >= vint i = l + 1, j = r;while(true){while(i <= j && arr[i] < arr[l])i ++;while(j >= i && arr[j] > arr[l])j --;if(i >= j) break;swap(arr, i, j);i ++;j --;}swap(arr, l, j);return j;}private void swap(int[] arr, int i, int j){int t = arr[i];arr[i] = arr[j];arr[j] = t;}
}

如果让调用java里库的方法,先 排序 后取值就更容易,不过  有点easy,面试时可能不让用

//时间复杂度:O(nlog⁡n),其中n是数组arr的长度。算法的时间复杂度即排序的时间复杂度。
//空间复杂度:O(log⁡n),排序所需额外的空间复杂度为 O(log⁡n)。
class Solution {public int[] smallestK(int[] arr, int k) {int[] vec = new int[k];Arrays.sort(arr); //调用Array.sort方法,给arr从小到大排序,然后取前k个数for (int i = 0; i < k; ++i) {vec[i] = arr[i];}return vec; //返回对应数组}
}
我在这个问题下提交如上的代码,大概需要 7ms 左右的时间:
但是,使用我们的 selectK 的方式,大概 2ms 就能搞定。
其实,O(n) 的复杂度和 O(nlogn) 的复杂度,在大多数时候,尤其是面试或者算
法竞赛中,差距并不大。但是,当数据规模上来以后,还是能看出来的。如果
有兴趣的同学,可以在自己的计算机上,测试一下对于 100 万甚至是 1000 万级
别的数据规模,看看而这是不是差距更明显?

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

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

相关文章

C++:基础语法

一、命名空间 在C/C中&#xff0c;变量、函数和后面要学到的类都是大量存在的&#xff0c;这些变量、函数和类的名称将都存在于全局作用域中&#xff0c;可能会导致很多冲突。使用命名空间的目的是对标识符的名称进行本地化&#xff0c; 以避免命名冲突或名字污染&#xff0c;n…

Hindawi暴雷出局,Frontiers却积极整改,能否摘掉“水刊”标签?

【SciencePub学术】自从3月Hindawi暴雷后&#xff0c;MDPI和Frontiers也深受牵连&#xff0c;因其发文量太过猖獗&#xff0c;国人占比高&#xff0c;自引率高等因素&#xff0c;这些出版社旗下的期刊均被贴上“水刊”标签。 上期&#xff0c;小编已经详细介绍了MDPI期刊的口碑…

鸿蒙 harmonyos 线程 并发 总结 async promise Taskpool woker(三)多线程并发 Worker

Worker Worker是与主线程并行的独立线程。创建Worker的线程称之为宿主线程&#xff0c;Worker自身的线程称之为Worker线程。创建Worker传入的url文件在Worker线程中执行&#xff0c;可以处理耗时操作但不可以直接操作UI。 Worker主要作用是为应用程序提供一个多线程的运行环境…

目标检测算法是指什么?

一、目标检测算法是指什么&#xff1f; 目标检测算法是计算机视觉领域的一个重要分支&#xff0c;它旨在识别和定位图像中的目标对象。以下是目标检测算法的相关内容&#xff1a; 目标检测的核心问题&#xff1a;目标检测需要解决的两个核心问题是“目标是什么”和“目标在哪里…

如何加盟共享wifi项目?了解套路有哪些?

自共享wifi项目推出在市场火爆后&#xff0c;各路资本都看到了该项目的广阔前景&#xff0c;纷纷开始研发程序&#xff0c;想要趁机分一杯羹。但对于普通人而言&#xff0c;独立研发程序显然不大现实&#xff0c;于是&#xff0c;共享wifi项目如何加盟便成为了绝大多数人最为关…

java异常的捕获

Java内置了一套异常处理机制&#xff0c;总是使用异常来表示错误。 一,使用try……catch来捕获错误 异常是一种class&#xff0c;因此它本身带有类型信息。异常可以在任何地方抛出&#xff0c;但只需要在上层捕获&#xff0c;这样就和方法调用分离了&#xff1a; try {Strin…

机器人自动驾驶时间同步进阶

0. 简介 之前时间同步也写过一篇文章介绍机器人&自动驾驶中的时间同步。在最近的学习中发现一些额外需要阐述学习的内容&#xff0c;这里就再次写一些之前没写到的内容。 1. NTP NTP 是网络时间协议&#xff0c;用来同步网络中各计算机时间的协议&#xff0c;把计算机的时…

[计算机效率] 网站推荐:网盘资源类

4.6 网盘资源类 在数字化时代&#xff0c;网盘资源搜索已成为我们日常生活和工作中不可或缺的一部分。无论是寻找工作资料、学习素材&#xff0c;还是娱乐资源&#xff0c;一个高效、可靠的网盘资源搜索网站都能为我们节省大量时间。今天&#xff0c;我将为大家推荐几个优秀的…

【C++类和对象】日期类的实现

&#x1f49e;&#x1f49e; 前言 hello hello~ &#xff0c;这里是大耳朵土土垚~&#x1f496;&#x1f496; &#xff0c;欢迎大家点赞&#x1f973;&#x1f973;关注&#x1f4a5;&#x1f4a5;收藏&#x1f339;&#x1f339;&#x1f339; &#x1f4a5;个人主页&#x…

华为公司战略规划和落地方法之五看三定工具解析【PPT图片】(内含超级福利)

导言 华为公司最厉害之处就是战略上的高举高打&#xff0c;“吹过的牛都实现了”。支撑华为公司战略从规划到落地的主要工具很多&#xff0c;其中“五看三定”是战略规划时最核心的方法之一。本资料将介绍五看三定的核心精髓。欢迎学习&#xff01; 本材料结合谢宁老师专著《华…

数据结构之顺序表的实现(C语言版)

Hello, 大家好&#xff0c;我是一代&#xff0c;今天给大家带来有关顺序表的有关知识 所属专栏&#xff1a;数据结构 创作不易&#xff0c;望得到各位佬们的互三呦 一.前言 1.首先在讲顺序表之前我们先来了解什么是数据结构 数据结构是由“数据”和“结构”两词组合⽽来。 什…

【Diffusion实战】训练一个diffusion模型生成蝴蝶图像(Pytorch代码详解)

上一篇Diffusion实战是确确实实一步一步走的公式&#xff0c;这回采用一个更方便的库&#xff1a;diffusers&#xff0c;来实现Diffusion模型训练。 Diffusion实战篇&#xff1a;   【Diffusion实战】训练一个diffusion模型生成S曲线&#xff08;Pytorch代码详解&#xff09;…