网课:第三章递归与分治思想---归并排序及相关运用

原理

把数组分为两半,两半分别排好序了,然后将两半数组排好序,形成一个有序数组。

时间复杂度:O(n*logn)

代码

复制代码
#include<bits/stdc++.h>
using namespace std;
int n;
int a[1000];
int b[1000];
void gb(int l,int r){
	if(l==r) return ;
	int mid=(l+r)/2;
	gb(l,mid);//分成两半
	gb(mid+1,r);
	int i=l,j=mid+1;
	int cnt=0;
	for(int k=l;k<=r;k++) {  //将两半重新排序成一个
		if( j>r || (a[i]<=a[j]&&i<=mid) )  //右半边的没有元素了或者左半边的有元素并且小于等于右半边的
			b[k]=a[i++];
		else 
			b[k]=a[j++];
	}
	for(int k=l;k<=r;k++) a[k]=b[k];
}
int main() {
	cin>>n;
	for(int i=0;i<n;i++) cin>>a[i];
	gb(0,n-1);//数组下标
	for(int i=0;i<n;i++) cout<<a[i]<<" ";
}

思路:所有排序都是在消灭逆序对,我们可以利用排序来写。

暴力解法:(1)两重for循环枚举i和j

(2)逆序对的个数等于冒泡排序中交换的次数

优化解法:归并排序(从小到大排)在重新排序两个部分时,右指针小于左指针时(左边部分还有元素),左指针之后到mid的元素就是右指针的逆序对

复制代码
#include<bits/stdc++.h>
using namespace std;
int n;
int a[1000];
int b[1000];
int cnt=0;
void gb(int l,int r){
	if(l==r) return ;
	int mid=(l+r)/2;
	gb(l,mid);
	gb(mid+1,r);
	int i=l,j=mid+1;
	int cnt=0;
	for(int k=l;k<=r;k++) {
		if( j>r || (a[i]<=a[j]&&i<=mid) ) 
			b[k]=a[i++];
		else {
			b[k]=a[j++];
			cnt+=mid-i+1;//逆序对
		}
			
	}
	for(int k=l;k<=r;k++) a[k]=b[k];
}
int main() {
	cin>>n;
	for(int i=0;i<n;i++) cin>>a[i];
	gb(0,n-1);
	cout<<cnt;
}
相关推荐
梵刹古音1 分钟前
【C语言】 跳转语句
c语言·开发语言·算法
liu****12 分钟前
29.路径类dp
c++·算法·acm
JMchen12313 分钟前
Android计算摄影实战:多帧合成、HDR+与夜景算法深度剖析
android·经验分享·数码相机·算法·移动开发·android-studio
阿猿收手吧!18 分钟前
【C++】C++模板特化:精准定制泛型逻辑
开发语言·c++·算法
智驱力人工智能32 分钟前
货车走快车道检测 高速公路安全治理的工程实践与价值闭环 高速公路货车占用小客车道抓拍系统 城市快速路货车违规占道AI识别
人工智能·opencv·算法·安全·yolo·目标检测·边缘计算
喵手42 分钟前
Python爬虫实战:电商实体消歧完整实战 - 从混乱店铺名到标准化知识库的工程化实现,一文带你搞定!
爬虫·python·算法·爬虫实战·零基础python爬虫教学·同名实体消除·从混乱店铺名到标准化知识库
weixin_452159551 小时前
C++与Java性能对比
开发语言·c++·算法
80530单词突击赢1 小时前
C++哈希表实现:开散列与闭散列详解
算法·哈希算法·散列表
Timmylyx05181 小时前
类欧几里得学习笔记
笔记·学习·算法
wangluoqi1 小时前
26.2.2练习总结
算法