【优选算法】(第二十四篇)

news/2024/10/6 18:28:26 标签: 算法, 数据结构, java, c++, leetcode, 排序算法, vscode

目录

归并排序(medium)

题目解析

讲解算法原理

编写代码

数组中的逆序对(hard)

题目解析

讲解算法原理

编写代码


归并排序(medium)

题目解析

1.题目链接:. - 力扣(LeetCode)

2.题目描述

给你⼀个整数数组nums,请你将该数组升序排列。
⽰例1:
输⼊:nums=[5,2,3,1]
输出:[1,2,3,5]
⽰例2:
输⼊:nums=[5,1,1,2,0,0]
输出:[0,0,1,1,2,5]

讲解算法原理

解法(归并排序):
算法思路:

归并排序的流程充分的体现了「分⽽治之」的思想,⼤体过程分为两步:

◦ 分:将数组⼀分为⼆为两部分,⼀直分解到数组的⻓度为 1 ,使整个数组的排序过程被分为
「左半部分排序」+「右半部分排序」;
◦ 治:将两个较短的「有序数组合并成⼀个⻓的有序数组」,⼀直合并到最初的⻓度。

编写代码

c++算法代码:
 

class Solution
{
 vector<int> tmp;
public:
 vector<int> sortArray(vector<int>& nums) 
 {
 tmp.resize(nums.size());
 mergeSort(nums, 0, nums.size() - 1);
 return nums;
 }
 void mergeSort(vector<int>& nums, int left, int right)
 {
 if(left >= right) return;
 // 1. 选择中间点划分区间
 int mid = (left + right) >> 1;
 // [left, mid] [mid + 1, right]
 // 2. 把左右区间排序
 mergeSort(nums, left, mid);
 mergeSort(nums, mid + 1, right);
 // 3. 合并两个有序数组
 int cur1 = left, cur2 = mid + 1, i = 0;
 while(cur1 <= mid && cur2 <= right)
 tmp[i++] = nums[cur1] <= nums[cur2] ? nums[cur1++] : 
nums[cur2++];
 // 处理没有遍历完的数组
 while(cur1 <= mid) tmp[i++] = nums[cur1++];
 while(cur2 <= right) tmp[i++] = nums[cur2++];
 // 4. 还原
 for(int i = left; i <= right; i++)
 nums[i] = tmp[i - left];
 }
};

java算法代码:

class Solution
{
 int[] tmp;
 public int[] sortArray(int[] nums) 
 {
 tmp = new int[nums.length];
 mergeSort(nums, 0, nums.length - 1);
 return nums;
 }
 public void mergeSort(int[] nums, int left, int right)
 {
 if(left >= right) return;
 // 1. 根据中间点划分区间
 int mid = (left + right) / 2;
 // [left, mid] [mid + 1, right]
 // 2. 先把左右区间排个序
 mergeSort(nums, left, mid);
 mergeSort(nums, mid + 1, right);
 // 3. 合并两个有序数组
 int cur1 = left, cur2 = mid + 1, i = 0;
 while(cur1 <= mid && cur2 <= right)
 tmp[i++] = nums[cur1] <= nums[cur2] ? nums[cur1++] : nums[cur2++];
 // 处理没有遍历完的数组
 while(cur1 <= mid) tmp[i++] = nums[cur1++];
 while(cur2 <= right) tmp[i++] = nums[cur2++];
 // 4. 还原
 for(int j = left; j <= right; j++) 
 nums[j] = tmp[j - left];
 }
}

数组中的逆序对(hard)

题目解析

1.题目链接:. - 力扣(LeetCode)

2.题目描述

在数组中的两个数字,如果前⾯⼀个数字⼤于后⾯的数字,则这两个数字组成⼀个逆序对。输⼊⼀个数组,求出这个数组中的逆序对的总数。
⽰例1:
输⼊:[7,5,6,4]输出:5

讲解算法原理

解法(利⽤归并排序的过程---分治):算法思路:
⽤归并排序求逆序数是很经典的⽅法,主要就是在归并排序的合并过程中统计出逆序对的数量,也就是在合并两个有序序列的过程中,能够快速求出逆序对的数量。
我们将这个问题分解成⼏个⼩问题,逐⼀破解这道题。
注意:默认都是升序,如果掌握升序的话,降序的归并过程也是可以解决问题的。• 先解决第⼀个问题,为什么可以利⽤归并排序?
如果我们将数组从中间划分成两个部分,那么我们可以将逆序对产⽣的⽅式划分成三组:• 逆序对中两个元素:全部从左数组中选择
• 逆序对中两个元素:全部从右数组中选择• 逆序对中两个元素:⼀个选左数组另⼀个选右数组根据排列组合的分类相加原理,三种种情况下产⽣的逆序对的总和,正好等于总的逆序对数量。
⽽这个思路正好匹配归并排序的过程:• 先排序左数组;
• 再排序右数组;• 左数组和右数组合⼆为⼀。
因此,我们可以利⽤归并排序的过程,先求出左半数组中逆序对的数量,再求出右半数组中逆序对的数量,最后求出⼀个选择左边,另⼀个选择右边情况下逆序对的数量,三者相加即可。
• 解决第⼆个问题,为什么要这么做?在归并排序合并的过程中,我们得到的是两个有序的数组。我们是可以利⽤数组的有序性,快速统计
出逆序对的数量,⽽不是将所有情况都枚举出来。
• 最核⼼的问题,如何在合并两个有序数组的过程中,统计出逆序对的数量?合并两个有序序列时求逆序对的⽅法有两种:
1. 快速统计出某个数前⾯有多少个数⽐它⼤;2. 快速统计出某个数后⾯有多少个数⽐它⼩;
⽅法⼀:快速统计出某个数前⾯有多少个数⽐它⼤通过⼀个⽰例来演⽰⽅法⼀:

假定已经有两个已经有序的序列以及辅助数组left=[5,7,9]right=[4,5,8]help=[],通过合并两个有序数组的过程,来求得逆序对的数量:
规定如下定义来叙述过程:
cur1遍历left数组,cur2遍历right数组ret记录逆序对的数量
第⼀轮循环:
left[cur1]>right[cur2],由于两个数组都是升序的,那么我们可以断定,此刻left数组中[cur1,2]区间内的3个元素均可与right[cur2]的元素构成逆序对,因此可以累加逆序对的数量ret+=3,并且将right[cur2]加⼊到辅助数组中,cur2++遍历下⼀个元素。
第⼀轮循环结束后:left=[5,7,9]right=[x,5,8]help=[4]ret=3cur1=0cur2=1
第⼆轮循环:
left[cur1]==right[cur2],因为right[cur2]可能与left数组中往后的元素构成逆序对,因此我们需要将left[cur1]加⼊到辅助数组中去,此时没有产⽣逆序对,不更新ret。
第⼆轮循环结束后:left=[x,7,9]right=[x,5,8]help=[4,5]ret=3cur1=1cur2=1
第三轮循环:
left[cur1]>right[cur2],与第⼀轮循环相同,此刻left数组中[cur1,2]区间内的2个元素均可与right[cur2]的元素构成逆序对,更新ret的值为ret+=2,并且将right[cur2]加⼊到辅助数组中去,cur2++遍历下⼀个元素。
第三轮循环结束后:left=[x,7,9]right=[x,x,8]help=[4,5,5]ret=5cur1=1cur2=2
第四轮循环:
left[cur1]<right[cur2],由于两个数组都是升序的,因此我们可以确定left[cur1]⽐right数组中的所有元素都要⼩。left[cur1]这个元素是不可能与right数组中的元素构成逆序对。因此,⼤胆
的将left[cur1]这个元素加⼊到辅助数组中去,不更细ret的值。第四轮循环结束后:left=[x,x,9]right=[x,x,8]help=[4,5,5,7]ret=5cur1=2cur2=2
第五轮循环:
left[cur1]>right[cur2],与第⼀、第三轮循环相同。此时left数组内的1个元素能与right[cur2]构成逆序对,更新ret的值,并且将right[cur2]加⼊到辅助数组中去。
第五轮循环结束后:left=[x,x,9]right=[x,x,x]help=[4,5,5,7,8]ret=6cur1=2cur2=2
处理剩余元素:• 如果是左边出现剩余,说明左边剩下的所有元素都是⽐右边元素⼤的,但是它们都是已经被计算过
的(我们以右边的元素为基准的),因此不会产⽣逆序对,仅需归并排序即可。• 如果是右边出现剩余,说明右边剩下的元素都是⽐左边⼤的,不符合逆序对的定义,因此也不需要
处理,仅需归并排序即可。
整个过程只需将两个数组遍历⼀遍即可,时间复杂度为O(N)。由上述过程我们可以得出⽅法⼀统计逆序对的关键点:
在合并有序数组的时候,遇到左数组当前元素>右数组当前元素时,我们可以通过计算左数组中剩余元素的⻓度,就可快速求出右数组当前元素前⾯有多少个数⽐它⼤,对⽐解法⼀中⼀个⼀个枚举逆序对效率快了许多。
⽅法⼆:快速统计出某个数后⾯有多少个数⽐它⼩依旧通过⼀个⽰例来演⽰⽅法⼆:
假定已经有两个已经有序的序列以及辅助数组left=[5,7,9]right=[4,5,8]help=[],通过合并两个有序数组的过程,来求得逆序对的数量:
规定如下定义来叙述过程:
cur1遍历left数组,cur2遍历right数组ret记录逆序对的数量
第⼀轮循环:
left[cur1]>right[cur2],先不要着急统计,因为我们要找的是当前元素后⾯有多少⽐它⼩的,这⾥虽然出现了⼀个,但是right数组中依旧还可能有其余⽐它⼩的。因此此时仅将right[cur2]加⼊到辅助数组中去,并且将cur2++。
第⼀轮循环结束后:left=[5,7,9]right=[x,5,8]help=[4]ret=0cur1=0cur2=1
第⼆轮循环:
left[cur1]==right[cur2],由于两个数组都是升序,这个时候对于元素left[cur1]来说,我们已经可以断定right数组中[0,cur2)左闭右开区间上的元素都是⽐它⼩的。因此此时可以统计逆序对的数量ret+=cur2-0,并且将left[cur1]放⼊到辅助数组中去,cur1++遍历下⼀个元素。
第⼆轮循环结束后:left=[x,7,9]right=[x,5,8]help=[4,5]ret=1cur1=1cur2=1
第三轮循环:
left[cur1]>right[cur2],与第⼀轮循环相同,直接将right[cur2]加⼊到辅助数组中去,cur2++遍历下⼀个元素。
第三轮循环结束后:left=[x,7,9]right=[x,x,8]help=[4,5,5]ret=1cur1=1cur2=2
第四轮循环:
left[cur1]<right[cur2],由于两个数组都是升序的,这个时候对于元素left[cur1]来说,我们依旧已经可以断定right数组中[0,cur2)左闭右开区间上的元素都是⽐它⼩的。因此此时可以统计逆序对的数量ret+=cur2-0,并且将left[cur1]放⼊到辅助数组中去,cur1++遍历下⼀个元素。
第四轮循环结束后:left=[9]right=[8]help=[4,5,5,7]ret=3cur1=2cur2=2
第五轮循环:
left[cur1]>right[cur2],与第⼀、第三轮循环相同。直接将right[cur2]加⼊到辅助数组中去,cur2++遍历下⼀个元素。
第五轮循环结束后:left=[x,x,9]right=[x,x,x]help=[4,5,5,7,8]ret=3cur1=2cur2=2
处理剩余元素:• 如果是左边出现剩余,说明左边剩下的所有元素都是⽐右边元素⼤的,但是相⽐较于⽅法⼀,逆序
对的数量是没有统计过的。因此,我们需要统计ret的值:
◦ 设左边数组剩余元素的个数为leave◦ ret+=leave*(cur2-0)
对于本题来说,处理剩余元素的时候,left数组剩余1个元素,cur2-0=3,因此ret需要类加上3,结果为6。与⽅法⼀求得的结果相同。
• 如果是右边出现剩余,说明右边剩下的元素都是⽐左边⼤的,不符合逆序对的定义,因此也不需要处理,仅需归并排序即可。
整个过程只需将两个数组遍历⼀遍即可,时间复杂度依旧为O(N)。
由上述过程我们可以得出⽅法⼆统计逆序对的关键点:
在合并有序数组的时候,遇到左数组当前元素<=右数组当前元素时,我们可以通过计算右数组已经遍历过的元素的⻓度,快速求出左数组当前元素后⾯有多少个数⽐它⼤。
但是需要注意的是,在处理剩余元素的时候,⽅法⼆还需要统计逆序对的数量。

编写代码
升序的版本

c++算法代码:

class Solution
{
 int tmp[50010];
public:
 int reversePairs(vector<int>& nums) 
 {
 return mergeSort(nums, 0, nums.size() - 1);
 }
 int mergeSort(vector<int>& nums, int left, int right)
 {
 if(left >= right) return 0;
 int ret = 0;
 // 1. 找中间点,将数组分成两部分
 int mid = (left + right) >> 1;
 // [left, mid][mid + 1, right]
 // 2. 左边的个数 + 排序 + 右边的个数 + 排序
 ret += mergeSort(nums, left, mid);
 ret += mergeSort(nums, mid + 1, right);
 // 3. ⼀左⼀右的个数
 int cur1 = left, cur2 = mid + 1, i = 0;
 while(cur1 <= mid && cur2 <= right) // 升序的时候 {
 if(nums[cur1] <= nums[cur2])
 {
 tmp[i++] = nums[cur1++];
 }
 else
 {
 ret += mid - cur1 + 1;
 tmp[i++] = nums[cur2++];
 }
 }
 // 4. 处理⼀下排序
 while(cur1 <= mid) tmp[i++] = nums[cur1++];
 while(cur2 <= right) tmp[i++] = nums[cur2++];
 for(int j = left; j <= right; j++)
 nums[j] = tmp[j - left];
 
 return ret;
 }
};

java算法代码:

class Solution
{ 
 int[] tmp;
 public int reversePairs(int[] nums) 
 {
 int n = nums.length;
 tmp = new int[n];
 return mergeSort(nums, 0, n - 1);
 }
 public int mergeSort(int[] nums, int left, int right)
 {
 if(left >= right) return 0;
 int ret = 0;
 // 1. 选择⼀个中间点,将数组划分成两部分
 int mid = (left + right) / 2;
 // [left, mid] [mid + 1, right]
 // 2. 左半部分的个数 + 排序 + 右半部分的个数 + 排序 ret += mergeSort(nums, left, mid);
 ret += mergeSort(nums, mid + 1, right);
 // 3. ⼀左⼀右的个数
 int cur1 = left, cur2 = mid + 1, i = 0;
 while(cur1 <= mid && cur2 <= right) // 升序版本 {
 if(nums[cur1] <= nums[cur2])
 {
 tmp[i++] = nums[cur1++];
 }
 else
 {
 ret += mid - cur1 + 1;
 tmp[i++] = nums[cur2++];
 }
 }
 // 4. 处理⼀下排序
 while(cur1 <= mid) tmp[i++] = nums[cur1++];
 while(cur2 <= right) tmp[i++] = nums[cur2++];
 for(int j = left; j <= right; j++)
 nums[j] = tmp[j - left];
 
 return ret;
 }
}
降序的版本

c++算法代码:

class Solution
{
 int tmp[50010];
public:
 int reversePairs(vector<int>& nums) 
 {
 return mergeSort(nums, 0, nums.size() - 1);
 }
 int mergeSort(vector<int>& nums, int left, int right)
 {
 if(left >= right) return 0;
 int ret = 0;
 // 1. 找中间点,将数组分成两部分
 int mid = (left + right) >> 1;
 // [left, mid][mid + 1, right]
 // 2. 左边的个数 + 排序 + 右边的个数 + 排序
 ret += mergeSort(nums, left, mid);
 ret += mergeSort(nums, mid + 1, right);
 // 3. ⼀左⼀右的个数
 int cur1 = left, cur2 = mid + 1, i = 0;
 while(cur1 <= mid && cur2 <= right) // 降序的版本 {
 if(nums[cur1] <= nums[cur2])
 {
 tmp[i++] = nums[cur2++];
 }
 else
 {
 ret += right - cur2 + 1;
 tmp[i++] = nums[cur1++];
 }
 }
 // 4. 处理⼀下排序
 while(cur1 <= mid) tmp[i++] = nums[cur1++];
 while(cur2 <= right) tmp[i++] = nums[cur2++];
 for(int j = left; j <= right; j++)
 nums[j] = tmp[j - left];
 
 return ret;
 }
};

java算法代码:

class Solution
{ 
 int[] tmp;
 public int reversePairs(int[] nums) 
 {
 int n = nums.length;
 tmp = new int[n];
 return mergeSort(nums, 0, n - 1);
 }
 public int mergeSort(int[] nums, int left, int right)
 {
 if(left >= right) return 0;
 int ret = 0;
 // 1. 选择⼀个中间点,将数组划分成两部分
 int mid = (left + right) / 2;
 // [left, mid] [mid + 1, right]
 // 2. 左半部分的个数 + 排序 + 右半部分的个数 + 排序 ret += mergeSort(nums, left, mid);
 ret += mergeSort(nums, mid + 1, right);
 // 3. ⼀左⼀右的个数
 int cur1 = left, cur2 = mid + 1, i = 0;
 while(cur1 <= mid && cur2 <= right) // 降序的版本 {
 if(nums[cur1] <= nums[cur2])
 {
 tmp[i++] = nums[cur2++];
 }
 else
 {
 ret += right - cur2 + 1;
 tmp[i++] = nums[cur1++];
 }
 }
 // 4. 处理⼀下排序
 while(cur1 <= mid) tmp[i++] = nums[cur1++];
 while(cur2 <= right) tmp[i++] = nums[cur2++];
 for(int j = left; j <= right; j++)
 nums[j] = tmp[j - left];
 
 return ret;
 }
}


http://www.niftyadmin.cn/n/5691991.html

相关文章

如何快速切换电脑的ip地址

在当今的数字化时代&#xff0c;IP地址作为网络身份的重要标识&#xff0c;其重要性日益凸显。无论是出于保护个人隐私的需要&#xff0c;还是为了访问特定的网络服务等&#xff0c;快速切换电脑的IP地址已成为许多用户的迫切需求。本文将为你介绍几种实用的方法&#xff0c;帮…

Mysql 索引底层数据结构和算法

目录 索引数据结构 Hash表 二叉树 红黑树 B树 B树 索引数据结构 索引&#xff08;index&#xff09;是帮助MySQL高效获取数据的一种有序数据结构。索引是存储到表空间中&#xff0c;当我们的 sql 中的where条件用到索引的时候&#xff0c;会在存储引擎层就过滤出数据来…

C++、Ruby和JavaScript

C C最初被称为带类的C, 兼容C的语法&#xff0c;此既是C得以流行的前提&#xff0c;也是C某些语法被捆绑的根源。C的来源于C语言的递增运算符&#xff0c;代表增加&#xff0c;意义为扩展。 C的历史 C类的设计思想来源于Simula. Simula为模拟的意思&#xff0c;被称为最早的面向…

计算机网络:物理层 —— 物理层下的传输媒体

文章目录 传输媒体导向性媒体同轴电缆双绞线光纤光纤分类中心波长光纤规格光纤的优缺点 非导向性媒体ISM 频段无线电波微波激光红外线可见光 传输媒体 传输媒体是计算机网络设备之间的物理通路&#xff0c;也称为传输介质或传输媒介&#xff0c;并不包含在计算机网络体系结构中…

计算机找不到vcomp140.dll,无法继续执行代码如何解决,有什么好的修复方法

1. vcomp140.dll 简介 1.1 定义 vcomp140.dll 是一个动态链接库&#xff08;DLL&#xff09;文件&#xff0c;它属于 Microsoft Visual C 2015 Redistributable Package 的一部分。该文件为应用程序提供了 OpenMP 并行框架所需的运行时支持&#xff0c;允许开发者编写并发和多…

Python知识点:如何使用SpaCy进行文本预处理与分析

开篇&#xff0c;先说一个好消息&#xff0c;截止到2025年1月1日前&#xff0c;翻到文末找到我&#xff0c;赠送定制版的开题报告和任务书&#xff0c;先到先得&#xff01;过期不候&#xff01; 如何使用SpaCy进行文本预处理与分析 在自然语言处理&#xff08;NLP&#xff09…

阿里巴巴开源的FastJson 1反序列化漏洞复现攻击保姆级教程

免责申明 本文仅是用于学习检测自己搭建的靶场环境有关FastJson1反序列化漏洞的原理和攻击实验,请勿用在非法途径上,若将其用于非法目的,所造成的一切后果由您自行承担,产生的一切风险和后果与笔者无关;本文开始前请认真详细学习《‌中华人民共和国网络安全法》‌及其所在…

自闭症寄宿学校陕西:提供综合发展的教育环境

在陕西这片古老而充满希望的土地上&#xff0c;有一所特殊的学校——星贝育园康复中心&#xff0c;它如同一座灯塔&#xff0c;照亮了无数自闭症儿童及其家庭前行的道路。这所全国规模较大的广泛性发育障碍全托寄宿制儿童康复训练机构&#xff0c;不仅以其专业的康复训练和独特…