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

原理

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

时间复杂度: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;
}
相关推荐
手写码匠3 分钟前
手写 LLM 安全护栏:从内容审核到越狱防御的完整实现
人工智能·深度学习·算法·aigc
luj_176813 分钟前
草酸与烟酸对消化及糖代谢的影响解析
服务器·c语言·开发语言·经验分享·算法
青风9724 分钟前
16-ADAPTRACK:基于自适应阈值的多目标跟踪匹配算法
人工智能·算法·目标跟踪
汤姆yu43 分钟前
macOS系统下Aider完整安装、配置与实战使用教程
大数据·人工智能·算法·macos·github·copilot
Sam09271 小时前
【AI 算法精讲 14】TF-IDF:词频与逆文档频率
人工智能·python·算法·ai
东华万里1 小时前
第31篇 数据结构入门:顺序表
数据结构·大学生专区
AI科技星1 小时前
拓扑生命系统确定性理论:基于32维流形的遗传密码起源与衰老动力学( 中英双语顶刊终稿·标准数学符号)
开发语言·网络·人工智能·算法·机器学习·乖乖数学·全域数学
编程圈子1 小时前
电机驱动开发学习18. SVPWM空间矢量调制算法详解与实现
驱动开发·学习·算法
大鱼>2 小时前
机器学习基础:从零理解核心概念与算法分类
算法·机器学习·分类
Vect__2 小时前
Go 数据结构 slice 深度剖析
开发语言·数据结构·golang