当前位置: 首页 > news >正文

leetcode - 分治思想

分治 - 快排

这里快排我们统一使用 数组分三块 和 随机产生基准值的方法实现排序

数组分三块:

. - 力扣(LeetCode)

整个思想即将数组按照基准值分为三个区间 , 具体实现: 三指针实现.  遍历指针 , 左区间右边界指针 , 右区间左边界指针

class Solution {//三指针法//格外注意边界值问题//[0,left - 1]  值为0//[left,right]  值为1//[right + 1,nums,length - 1] 值为2public void sortColors(int[] nums) {int left = 0;int right = nums.length - 1;int i = 0;while(i <= right) {if(nums[i] == 0) swap(left ++,i ++,nums);else if(nums[i] == 1) i ++;else swap(right --,i,nums);//注意这里右边交换时i不用++,}}private void swap(int left,int right,int[] nums) {int cur = nums[left];nums[left] = nums[right];nums[right] = cur;}
}

过程

为什么要使用随机数?   基于随机数产生基准值而进行快排时,时间复杂度最为接近O(Logn)

class Solution {public int[] sortArray(int[] nums) {sort(0 , nums.length - 1 , nums);return nums;}private void sort(int left , int right , int[] nums){//递归结束条件if(left >= right) return;int random = new Random().nextInt(right - left + 1);int flag = nums[random + left];//确定基准值int leftFlag = left;int rightFlag = right;int cur = left;while(cur <= rightFlag){if(nums[cur] == flag) cur ++;else if(nums[cur] > flag) swap(nums , cur , rightFlag --);else swap(nums , cur++ , leftFlag ++);}//循环结束时,leftModel - 1为左区间最后一个元素 , leftModel + 1为右区间最后一个元素//递归sort(left , leftFlag - 1 ,nums);sort(rightFlag + 1 , right , nums);}private void swap(int[] nums , int l1 , int l2) {int cur = nums[l1];nums[l1] = nums[l2];nums[l2] = cur;}}

快速选择算法:

. - 力扣(LeetCode)    数组中第k个最大元素  , 要求时间复杂度O(n). 

这道题比较容易想到的解法是 借助堆.    但是题目中要求时间复杂度为On , 堆就不满足要求了.

时间复杂度为O(n). 我们就要想到快速选择排算法

原理:  快排数组分三段的基础上不去对每一段进行排序,而是通过三个区间的元素数量判断目标元素在哪一个区间进行分类讨论:

class Solution {//快速选择算法//基于快排数组分三块的思想:分三块后通过分类确定目标范围private static int dest;public int findKthLargest(int[] nums, int k) {Solution s = new Solution();s.quickChoose(0 , nums.length - 1 , nums , k);return dest;}private void quickChoose(int left , int right , int[] nums , int k){if(right <= left) {dest = nums[left];//这里不要忘记赋值return;}int flag = nums[new Random().nextInt(right - left + 1) + left];System.out.println(left+":" + right+":" +flag);int l = left;int r = right;int cur = left;while(cur <=  r){if(nums[cur] == flag) cur ++;else if(nums[cur] > flag) swap(nums , r -- , cur);//注意右边换过来的是一个新值,cur不进行++else if(nums[cur] < flag) swap(nums , l ++ , cur ++);}//分类讨论 , 根据区间内元素的数量确定下一步计划//1: 目标元素在大于基准值的区间if(right - r >= k) quickChoose(r + 1 , right ,nums , k);//2: 目标元素等于基准值,直接进行返回else if(right - l + 1 >= k) {dest = flag;return;}//3:目标元素在小于基准值的区间else quickChoose(left , l - 1, nums , k - (right - l + 1));}private void swap(int[] nums , int num1 , int num2) {int cur = nums[num1];nums[num1] = nums[num2];nums[num2] = cur;}}

. - 力扣(LeetCode)   库存管理: 选出库存最小的cnt个商品

class Solution {private int str;private int end;//使用快速选择排序public int[] inventoryManagement(int[] stock, int cnt) {quickSort(stock , 0 , stock.length - 1 , cnt);int[] dest = new int[cnt];for(int i = 0;i < cnt;i ++) {dest[i] = stock[i];}return dest;}private void quickSort(int[] nums , int left , int right , int cnt) {if(left >= right) return;int l = left;int r = right;int cur =  left;int flag = nums[new Random().nextInt(right - left + 1) + left];while(cur <= r){if(nums[cur] == flag) cur ++;else if(nums[cur] > flag) swap(r -- , cur , nums);else swap(l ++ , cur++ , nums);}//分类讨论if(l - left > cnt) quickSort(nums , left , l - 1 , cnt);else if(r - left + 1 >= cnt) return;else quickSort(nums , r + 1, right , cnt - (r - left + 1));}private void swap(int num1 , int num2 , int[] nums) {int temp = nums[num1];nums[num1] = nums[num2];nums[num2] = temp;}
}

分治 - 归并

核心思想:   将一个问题分成若干个规模较小的子问题,分别求解后,再将子问题的结果合并,得到原问题的解。归并排序的核心步骤分为两个部分:“分”和“合”。

归并排序

:. - 力扣(LeetCode)

 public int[] sortArray(int[] nums) {sort(nums , 0 , nums.length - 1);return nums;}private void sort(int[] nums , int left , int right) {if(left >= right) return;//不断递归,将数组分割成俩部分,对两部分别再进行分割,直到每一部只剩下一个元素 , 俩俩进行合并int mid = left + (right - left) / 2;sort(nums , left , mid);sort(nums , mid + 1 , right);sortHelper(nums , left , mid , right);//合并的逻辑}//合并的俩个数组分别已经有序 , 建立一个新数组,设置俩个指针对俩个数组进行比较赋值到新数组上,使新数组比较//不要直接再原数组上进行比较交换 , 会打乱数组的有序性 , 将问题复杂化private void sortHelper(int[] nums , int left , int mid , int right) {int l = left;int r = mid + 1;int[] temp = new int[right - left + 1];int cur = 0;while(l <= mid && r <= right) {while(l <= mid && nums[l] <= nums[r]) temp[cur ++] = nums[l ++];while(r <= right && nums[r] <= nums[l]) temp[cur ++] = nums[r ++];}if(l <= mid){while(l <= mid)temp[cur ++] = nums[l ++];}if(r <= right) {while(r <= right) temp[cur ++] = nums[r ++];}for(int i = 0;i < temp.length;i ++) {nums[left + i] = temp[i];}}

优化 : 在进行合并时创建一个全局的,与原数组等大小的临时数组 , 这样就能避免每次进行合并时创建数组的开销.

例题

统计逆序对:  . - 力扣(LeetCode)

在上述归并的逻辑上加上 数组间统计逆序对 的逻辑即可

class Solution {//在进行数组之间合并排序之前加上一个统计数组间逆序队的逻辑`即可public int reversePairs(int[] record) {return sort(record , 0 , record.length - 1);}private int sort(int[] nums  , int left , int right) {if(left >= right) return 0;int mid = left + (right - left) / 2;int num1 = sort(nums , left , mid);int num2 = sort(nums , mid + 1 , right);int num3 = sortHelper(nums , left , mid , right);return (num1 + num2 + num3);}private int sortHelper(int[] nums ,int left ,int mid , int right){int l = left;int r = mid + 1;int count = 0;int[] model = new int[right - left + 1];int cur = 0;while(l <= mid && r <= right){while(l <= mid && nums[l] <= nums[r]) model[cur ++] = nums[l ++];while(r <= right && nums[r] < nums[l]){count += mid - l + 1;//统计逆序对; 因为合并的俩个数组已经分别有序,所以只需要统计数组间即可model[cur ++] = nums[r ++];}}while(l <= mid) model[cur ++] = nums[l ++];while(r <= right) model[cur ++] = nums[r ++];for(int i = 0;i < model.length;i ++) {nums[left + i] = model[i];}return count;}
}

右侧小于自己的数:. - 力扣(LeetCode)

首先:这道题使用暴力求解一定会超时

这道题进行归并的问题是: 归并排序后会进行元素交换, 交换后元素顺序被打乱  , 我们在进行合并数组时统计出来右侧小于当前元素的数后不知道这个元素的原始下标 .  所以这道题相比于上面求逆序对 , 需要多维护一个存储元素下标的数组.  原数组元素顺序改变时 , 这个存储下标的数组也随之进行改变 , 这样,我们就能通过nums中元素的下标拿到当前元素的初始位置 .

class Solution {private int[] indexMoel;//存储每个元素的原始位置 , 随着nums元素交换而交换,nums元素下标对应到indexModel上就是初始元素的位置,通过初始元素位置在dest上进行数量更新private int[] dest;//存储目标元素private int[] numsTemp;//使用全局数组进行合并,避免每次合并创建临时数组的开销private int[] indexTemp;public List<Integer> countSmaller(int[] nums) {indexMoel = new int[nums.length];numsTemp = new int[nums.length];indexTemp = new int[nums.length];dest = new int[nums.length];for(int i = 0;i < nums.length;i ++) {indexMoel[i] = i;}sort(nums , 0 , nums.length - 1);List<Integer> list = new ArrayList<>();for(int i = 0;i < dest.length;i ++) {list.add(dest[i]);}return list;}private void sort(int[] nums , int left , int right) {if(left >= right) return;int mid = left + (right - left) / 2;sort(nums , left , mid);sort(nums , mid + 1 , right);sortHelper(nums , left , mid , right);}private void sortHelper(int[] nums , int left , int mid , int right) {int l = left;int r = mid + 1;int cur1 = 0;int cur2 = 0;//降序排序 , 更好统计while(l <= mid && r <= right) {if(nums[l] > nums[r]) {dest[indexMoel[l]] += right - r + 1;//通过indexmodel拿到当前元素的初始位置进行统计numsTemp[cur1 ++] = nums[l];indexTemp[cur2 ++] = indexMoel[l ++];//indexModel随着nums的交换而交换}else {numsTemp[cur1 ++] = nums[r];indexTemp[cur2 ++] = indexMoel[r ++];}}while(l <= mid){numsTemp[cur1 ++] = nums[l];indexTemp[cur2 ++] = indexMoel[l ++];}while(r <= right){numsTemp[cur1 ++] = nums[r];indexTemp[cur2 ++] = indexMoel[r ++];}for(int i = 0;i < cur1;i ++) {nums[i + left] = numsTemp[i];indexMoel[i + left] = indexTemp[i]; }}
}

相关文章:

  • 北京网站建设多少钱?
  • 辽宁网页制作哪家好_网站建设
  • 高端品牌网站建设_汉中网站制作
  • 前后端数据交互 笔记03(get和post方法)
  • hku-mars雷达相机时间同步方案-软件驱动(MID360与海康MV-CB060-10UMUC-S)
  • [Redis] Redis中的Hash类型和List类型
  • 【CTF Web】BUUCTF BUU UPLOAD COURSE 1 Writeup(文件上传+PHP+文件包含漏洞)
  • 泛微E10产品二开
  • OrionX GPU算力池助力AI OCR场景应用
  • 鹏哥C语言36-37---循环/分支语句练习(折半查找算法)
  • CleanClip for Mac 剪切板 粘贴工具 历史记录 安装(保姆级教程,新手小白轻松上手)
  • (SERIES12)DM性能优化
  • Redis如何实现分布式锁
  • Kali Linux 2024.3 发布,包含新黑客工具
  • 挑战力扣高难度算法、数据库题
  • 基于Arduino Uno开发板实现PWM呼吸灯的设计方案
  • C语言-结构体-详解
  • 【数据结构与算法 | 灵神题单 | 自顶向下DFS篇】力扣1022,623
  • $translatePartialLoader加载失败及解决方式
  • 2017 前端面试准备 - 收藏集 - 掘金
  • Create React App 使用
  • HTTP中的ETag在移动客户端的应用
  • JavaSE小实践1:Java爬取斗图网站的所有表情包
  • PAT A1120
  • Python3爬取英雄联盟英雄皮肤大图
  • python大佬养成计划----difflib模块
  • 记录一下第一次使用npm
  • 设计模式 开闭原则
  • 时间复杂度与空间复杂度分析
  • 数据仓库的几种建模方法
  • 限制Java线程池运行线程以及等待线程数量的策略
  • 小程序01:wepy框架整合iview webapp UI
  • 一个SAP顾问在美国的这些年
  • 你对linux中grep命令知道多少?
  • raise 与 raise ... from 的区别
  • ​​​​​​​Installing ROS on the Raspberry Pi
  • "无招胜有招"nbsp;史上最全的互…
  • # Swust 12th acm 邀请赛# [ K ] 三角形判定 [题解]
  • #NOIP 2014# day.2 T2 寻找道路
  • $(function(){})与(function($){....})(jQuery)的区别
  • (2024,RWKV-5/6,RNN,矩阵值注意力状态,数据依赖线性插值,LoRA,多语言分词器)Eagle 和 Finch
  • (70min)字节暑假实习二面(已挂)
  • (C#)一个最简单的链表类
  • (pt可视化)利用torch的make_grid进行张量可视化
  • (二) Windows 下 Sublime Text 3 安装离线插件 Anaconda
  • (二十五)admin-boot项目之集成消息队列Rabbitmq
  • (含react-draggable库以及相关BUG如何解决)固定在左上方某盒子内(如按钮)添加可拖动功能,使用react hook语法实现
  • (未解决)jmeter报错之“请在微信客户端打开链接”
  • (五)MySQL的备份及恢复
  • (限时免费)震惊!流落人间的haproxy宝典被找到了!一切玄妙尽在此处!
  • (心得)获取一个数二进制序列中所有的偶数位和奇数位, 分别输出二进制序列。
  • (学习日记)2024.03.25:UCOSIII第二十二节:系统启动流程详解
  • (转)iOS字体
  • (转)mysql使用Navicat 导出和导入数据库
  • (自用)learnOpenGL学习总结-高级OpenGL-抗锯齿
  • *2 echo、printf、mkdir命令的应用
  • .net refrector
  • .net Stream篇(六)