优选算法的妙思之流:分治——归并专题

专栏:算法的魔法世界

个人主页:手握风云

目录

一、归并排序

二、例题讲解

2.1. 排序数组

2.2. 交易逆序对的总数

2.3. 计算右侧小于当前元素的个数

2.4. 翻转对


一、归并排序

        归并排序也是采用了分治的思想,将数组划分为多个长度为1的子数组进行排序,再把多个子数组合并为最终的数组。

二、例题讲解

2.1. 排序数组

        我们再来回顾一下归并排序的大体思路:以数组的中间值将数组划分为两个子数组,以此再继续划分,直至将数组划分为里面只有一个元素,就可以向上返回。当左边排完后,再去排右边,然后再将两个子数组进行合并,直到合并为原来的数组长度。

        完整代码实现:

class Solution {public int[] sortArray(int[] nums) {MergeSort(nums, 0, nums.length - 1);return nums;}private void MergeSort(int[] nums, int left, int right) {if (left >= right)return;//根据中间点划分左右子数组int mid = (left + right) / 2;MergeSort(nums, left, mid);MergeSort(nums, mid + 1, right);//合并两个有序数组int[] tmp = new int[right - left + 1];int p1 = left, p2 = mid + 1, i = 0;while (p1 <= mid && p2 <= right) {tmp[i++] = nums[p1] <= nums[p2] ? nums[p1++] : nums[p2++];}while (p1 <= mid) {tmp[i++] = nums[p1++];}while (p2 <= right) {tmp[i++] = nums[p2++];}//还原for (int j = left; j <= right; j++) {nums[j] = tmp[j - left];}}
}

2.2. 交易逆序对的总数

        暴力解法,利用两层for循环,先固定其中一个数,再让其他数与这个数进行比较,如果小,就让计数器++。

class Solution {public int reversePairs(int[] record) {int count = 0;for(int i = 0;i < record.length;i ++){for(int j = i + 1;j < record.length;j++){if(record[i] > record[j]) count++;}}return count;}
}

        第二种解法,将数组划分为两块,先找出左区域的逆序对,再找出右区域的逆序对,最后再一左一右随机挑一个数找出逆序对。但这样的解法本质上还是一个暴力枚举。我们接着来及逆行优化,我们先找完左区域的逆序对,然后对左区域进行排序;找完右区域的逆序对,然后对右区域进行排序。我们在左右区域里面随机挑数的时候是不会影响逆序对的数量。

        当我们在左右区域里面分别寻找逆序对时,如果数组长度较大,那么我们还可以再接着划分,继续按照上面的思路来找出逆序对的总数,这个部分就可以在递归中完成,所以我们在处理一左一右时也可以排个序。

        接下来是查找子数组里面逆序对的数目,如下图所示,p1左侧是子数组中较小的元素,p2左侧也是子数组中较小的元素。我们先固定p2这个数,接下来就是在[left,p1]这段区间里面寻找比p2大的元素。如果p1所指的元素比p2所指的元素小或者等于,那么就让p1++;如果p1所指的元素比p2所指的元素大,那么p1后面的元素就都是比p2大,就可以快速的统计出数目,然后再让p2++。

        完整代码实现:

class Solution {int[] tmp;public int reversePairs(int[] record) {int n = record.length;tmp = new int[n];return MergeSort(record, 0, n - 1);}private int MergeSort(int[] nums, int left, int right) {if (left >= right) return 0;int ret = 0;//中点元素int mid = (left + right) / 2;//左半部分的数目+右半部分的数目ret += MergeSort(nums, left, mid);ret += MergeSort(nums, mid + 1, right);//一左一右的数目int p1 = left, p2 = mid + 1, i = 0;while (p1 <= mid && p2 <= right) {if (nums[p1] <= nums[p2]) {tmp[i++] = nums[p1++];} else {ret += mid - p1 + 1;tmp[i++] = nums[p2++];}}while (p1 <= mid) tmp[i++] = nums[p1++];while (p2 <= right) tmp[i++] = nums[p2++];for (int j = left; j <= right; j++) {nums[j] = tmp[j - left];}return ret;}
}

2.3. 计算右侧小于当前元素的个数

        这道题要我们求出一个数组元素右边有多少个数比它小,我们可以仿照上一题的思路,只不过这道题要把数组降序排列。

        把数组分成两部分,先找出左区域里面比自身小的个数,再找出右区域比自身小的个数,再一左一右去寻找个数。因为数组是降序排列的,所以p1、p2左边都是相对较大的元素。如果nums[p1]<=nums[p2],则p2++;如果nums[p1]>nums[p2],因为我们最终是要返回一个顺序表,所以我们要用该元素对应的位置来统计结果(right-p2+1)。但是经过归并排序后,数组下标已经乱了,下一步就是要求数组元素对应的原始下标。

        我们可以使用哈希思想来解决数组下标与元素的动态绑定,如果数组里面有重复元素,使用哈希表就很难。当对数组进行排序时,数组下标也要随着变换。

        完整代码实现:

class Solution {int[] ret;int[] index;//标记原始下标int[] tmpIndex;//用于合并时的数组int[] tmpNum;public List<Integer> countSmaller(int[] nums) {int n = nums.length;ret = new int[n];index = new int[n];tmpIndex = new int[n];tmpNum = new int[n];//初始化index数组for (int i = 0; i < n; i++) {index[i] = i;}Mergesort(nums,0,n - 1);List<Integer> res = new ArrayList<>();for(int x : ret)res.add(x);return res;}private void Mergesort(int[] nums, int left, int right) {if(left >= right) return;//根据中间元素划分区间int mid = (left + right) / 2;//处理左右区间Mergesort(nums,left,mid);Mergesort(nums,mid + 1,right);//合并int p1 = left,p2 = mid + 1,i = 0;while(p1 <= mid && p2 <= right){//降序if(nums[p1] <= nums[p2]){tmpNum[i] = nums[p2];tmpIndex[i++] = index[p2++];} else {ret[index[p1]] += right - p2 + 1;tmpNum[i] = nums[p1];tmpIndex[i++] = index[p1++];}}//处理剩余的排序while(p1 <= mid){tmpNum[i] = nums[p1];tmpIndex[i++] = index[p1++];}while(p2 <= right){tmpNum[i] = nums[p2];tmpIndex[i++] = index[p2++];}for (int j = left; j <= right; j++) {nums[j] = tmpNum[j - left];index[j] = tmpIndex[j - left];}}
}

2.4. 翻转对

        我们依然可以按照逆序对那道题,将数组划分为两块,求出左区域、右区域以及一左一右的翻转对的和。但是逆序对那道题是1:1进行比较的,而这道题需要前面的元素大于后面元素的2倍,就不能按照归并排序的思路来解决。所以我们先计算翻转对的数目,我们可以有两种思路:1.计算当前元素后面,有多少元素的2倍比该数小;2.计算当前元素后面,有多少元素的一半比该数大。

        我们先来讲下方法1(按照降序排列,没有边界):利用数组有序的特性计算翻转对。如果nums[p1]<=nums[p2],p2向后移动,直到nums[p1]>nums[p2]时,就可以统计出数目来。如果我们让p2回退,那么时间复杂度就会达到n^{2}级别。因为数组是降序的,当p1右移时,所指向的元素也会变小,没必要再让p2回退,就可以构成同向双指针了。方法2也同理,如果nums[p1]>= nums[p2],就让p1向右移动,直到nums[p1]<nums[p2],再让p2向右移动,p1同样不用回退。

        完整代码实现:

class Solution {int[] tmp;public int reversePairs(int[] nums) {int n = nums.length;tmp = new int[n];return Mergesort(nums,0,n - 1);}private int Mergesort(int[] nums, int left, int right) {if(left >= right) return 0;int ret = 0;int mid = (left + right) / 2;ret += Mergesort(nums,left,mid);ret += Mergesort(nums,mid + 1,right);int p1 = left,p2 = mid + 1,i = left;while(p1 <= mid) {while (p2 <= right && nums[p2] >= nums[p1] / 2.0) p2++;if(p2 > right)break;ret += right - p2 + 1;p1++;}p1 = left;p2 = mid + 1;while(p1 <= mid && p2 <= right){tmp[i++] = (nums[p1] <= nums[p2]) ? nums[p2++] : nums[p1++];}while(p1 <= mid) tmp[i++] = nums[p1++];while(p2 <= right) tmp[i++] = nums[p2++];for (int j = left; j <= right; j++) {nums[j] = tmp[j];}return ret;}
}

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

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

相关文章

C语言查漏补缺:基础篇

1.原理 C语言是一门编译型计算机语言&#xff0c;要编写C代码&#xff0c;C源代码文本文件本身无法直接执行&#xff0c;必须通过编译器翻译和链接器的链接&#xff0c;生成二进制的可执行文件&#xff0c;然后才能执行。这里的二进制的可执行文件就是我们最终要形成的可执行程…

TPS入门DAY02 服务器篇

1.创建空白插件 2.导入在线子系统以及在线steam子系统库 MultiplayerSessions.uplugin MultiplayerSessions.Build.cs 3.创建游戏实例以及初始化会话创建流程 创建会话需要的函数&#xff0c;委托&#xff0c;委托绑定的回调&#xff0c;在线子系统接口绑定某一个委托的控制其…

产品经理课程

原型工具 一、土耳其机器人 这个说法来源于 1770 年出现的一个骗局&#xff0c;一个叫沃尔夫冈冯肯佩伦&#xff08;Wolfgang von Kempelen&#xff09;的人为了取悦奥地利女皇玛丽娅特蕾莎&#xff08;Maria Theresia&#xff09;&#xff0c;“制造”了一个会下国际象棋的机…

nginx中的limit_req 和 limit_conn

在 Nginx 中&#xff0c;limit_req 和 limit_conn 是两个用于限制客户端请求的指令&#xff0c;它们分别用于限制请求速率和并发连接数。 limit_req limit_req 用于限制请求速率&#xff0c;防止客户端发送过多请求影响服务器性能。它通过 limit_req_zone 指令定义一个共享内存…

基于winform的串口调试助手

目录 一、串口助手界面设计 1.1 串口配置 1.2 接收配置 1.3 发送配置 1.4 接收窗口和发送窗口 1.5 状态显示窗口 1.6 串口通讯控件 二、程序编写 2.1 端口号自动识别并显示在端口号下拉框 功能说明&#xff1a; 2.2 波特率下拉框显示 2.3 数据位下拉框显示 2.4 校…

Docker基础2

如需转载&#xff0c;标记出处 本次我们将下载一个 Docker 镜像&#xff0c;从镜像中启动容器 上一章&#xff0c;安装 Docker 时&#xff0c;获得两个主要组件&#xff1a; Docker 客户端 Docker 守护进程&#xff08;有时称为“服务器”或“引擎”&#xff09; 守护进程实…

Rocketmq2

一、生产者端防丢失 1. 发送方式选择 同步发送&#xff1a;使用 send() 方法&#xff0c;等待 Broker 确认响应&#xff08;SendResult&#xff09;&#xff0c;确保消息已成功发送。异步发送&#xff1a;使用 sendAsync() 方法并设置回调函数&#xff0c;处理发送成功 / 失败…

RabbitMQ详解,RabbitMQ是什么?架构是怎样的?

目录 一,RabbitMQ是什么? 二,RabbitMQ架构 2.1 首先我们来看下RabbitMQ里面的心概念Queue是什么? 2.2 交换器Exchange 2.3 RabbitMQ是什么? 2.4 重点看下优先级队列是什么? 三,RabbitMQ集群 3.1 普通集群模式 3.2 镜像队列集群 一,RabbitMQ是什么? 假设我们程序…

【一步步开发AI运动APP】六、运动计时计数能调用

之前我们为您分享了【一步步开发AI运动小程序】开发系列博文&#xff0c;通过该系列博文&#xff0c;很多开发者开发出了很多精美的AI健身、线上运动赛事、AI学生体测、美体、康复锻炼等应用场景的AI运动小程序&#xff1b;为了帮助开发者继续深耕AI运动领域市场&#xff0c;今…

MySQL——DQL的多表查询

一、交叉连接 标准语法&#xff1a;select * from 表1 cross join 表2 where 表1.公共列 表2.公共列; 简单语法&#xff1a;select * from 表1 , 表2 where 表1.公共列 表2.公共列; 公共列&#xff1a;两张表具有相同含义的列&#xff0c;不是列名一样。 …

【Linux内核】如何更加优雅阅读Linux内核源码(vscode)

1. 前言 因为已经习惯在Ubuntu下进行嵌入式工作开发&#xff0c;但Linux源码在Source Insight下进行阅读&#xff0c;一直很苦恼Linux/Windows来回切换的开发方式&#xff0c;当前发现可以通过 vscode clangd(扩展组件) 方式进行更好的内核源码阅读。 2. 环境 操作系统&…

21.OpenCV获取图像轮廓信息

OpenCV获取图像轮廓信息 在计算机视觉领域&#xff0c;识别和分析图像中的对象形状是一项基本任务。OpenCV 库提供了一个强大的工具——轮廓检测&#xff08;Contour Detection&#xff09;&#xff0c;它能够帮助我们精确地定位对象的边界。这篇博文将带你入门 OpenCV 的轮廓…

LETTERS(DFS)

【题目描述】 给出一个rowcolrowcol的大写字母矩阵&#xff0c;一开始的位置为左上角&#xff0c;你可以向上下左右四个方向移动&#xff0c;并且不能移向曾经经过的字母。问最多可以经过几个字母。 【输入】 第一行&#xff0c;输入字母矩阵行数RR和列数SS&#xff0c;1≤R,S≤…

Day2-2:前端项目uniapp壁纸实战

再在wallpaper新建一个目录components 在components下新建组件common-title 记得点击创建同名目录 在index加 <view class"select"><common-title></common-title></view> 图片换了下&#xff0c;原来的有点丑&#xff0c;图片可按自己喜欢…

其他 vector 操作详解(四十)

介绍 除去向 vector 添加元素&#xff08;如 push_back&#xff09;之外&#xff0c;vector 还提供了许多其他操作&#xff0c;这些操作大多与 string 的操作类似。通过掌握这些操作&#xff0c;我们可以方便地查询、修改和比较 vector 中的元素&#xff0c;从而构建灵活、高效…

【Leetcode 每日一题】368. 最大整除子集

问题背景 给你一个由 无重复 正整数组成的集合 n u m s nums nums&#xff0c;请你找出并返回其中最大的整除子集 a n s w e r answer answer&#xff0c;子集中每一元素对 ( a n s w e r [ i ] , a n s w e r [ j ] ) (answer[i], answer[j]) (answer[i],answer[j]) 都应当…

python基础-13-处理excel电子表格

文章目录 【README】【13】处理Excel电子表格【13.1】Excel文档【13.2】安装openpyxl模块【13.3】读取Excel文档【13.3.1】使用openpyxl模块打开excel文档【13.3.2】从工作簿取得工作表【13.3.3】从工作表sheet获取单元格cell【13.3.5】从表中获取行和列【13.3.6】工作簿、工作…

ABS函数c++

简介&#xff1a; abs 函数用于计算一个数的绝对值&#xff0c;在 C 中它继承自 C 语言的标准库&#xff0c;其历史可以追溯到早期的 C 语言发展历程&#xff0c;以下是详细介绍&#xff1a; 早期编程语言的需求 在计算机编程的早期阶段&#xff0c;处理数学运算就是一项基本…

闭环SOTA!北航DiffAD:基于扩散模型实现端到端自动驾驶「多任务闭环统一」

端到端自动驾驶目前是有望实现完全自动驾驶的一条有前景的途径。然而&#xff0c;现有的端到端自动驾驶系统通常采用主干网络与多任务头结合的方式&#xff0c;但是它们存在任务协调和系统复杂度高的问题。为此&#xff0c;本文提出了DiffAD&#xff0c;它统一了各种驾驶目标并…

整车CAN网络和CANoe

车载网络中主要包含有Can网络,Lin网络,FlexRay,Most,以太网。 500kbps:500波特率,表示的数据传输的速度。表示的是最大的网速传输速度。也就是每秒 500kb BodyCan车身Can InfoCan娱乐信息Can 车身CAN主要连接的是ESB电动安全带 ADB自适应远光灯等 PTCan动力Can 底盘Can