归并排序求逆序对的数量
一、题目来源
二、题目描述
给定一个长度为
逆序对的定义如下:对于数列的第
输入格式**
第一行包含整数
第二行包含
输出格式
输出一个整数,表示逆序对的个数。
数据范围
数列中的元素的取值范围
输入样例:
6
2 3 4 5 6 1
输出样例:
5
三、算法思路
本题基于归并排序的过程。
思路如下:
-
先将区间分为两段
-
满足逆序对条件的两个点有如下三种情况:
- 两个点都在左区间
- 两个点都在右区间
- 一个点在左边区间,一个点在右边区间
-
不断的递归排序的过程中,上述
情况都会转化为 的情况,所以答案直接加上左区间排序和右区间排序即可 -
由于归并之后两个区间都是有序的,所以如果当前左区间的数大于当前右区间的数,则左区间剩下的所有数都大于当前右区间的数,即满足逆序对的条件,
res += mid - i + 1;
- 如果直接暴力解题,从后往前遍历,时间复杂度
,采用归并排序解题时间复杂度为 。 - 注意思考上述三种情况,在不断划分区间的过程中,最后都会转化为第三种情况。
- 数据量很大可能会爆
int
,所以用long long
来存答案。
四、源代码
#include <iostream>
using namespace std;
const int N = 100010;
typedef long long LL;
int n;
int a[N];
int tmp[N];
LL merge_sort(int a[], int l, int r)
{
if (l >= r) return 0;
int mid = (l + r) >> 1;
LL res = merge_sort(a, l, mid) + merge_sort(a, mid + 1, r);
int k = 0, i = l, j = mid + 1;
while (i <= mid && j <= r)
if (a[i] <= a[j]) tmp[k ++ ] = a[i ++ ];
else res += mid - i + 1, tmp[k ++ ] = a[j ++ ];
while (i <= mid) tmp[k ++ ] = a[i ++ ];
while (j <= r) tmp[k ++ ] = a[j ++ ];
for (int i = l, j = 0; i <= r; ++i, ++j) a[i] = tmp[j];
return res;
}
int main()
{
cin >> n;
for (int i = 0; i < n; ++i) cin >> a[i];
cout << merge_sort(a, 0, n - 1) << endl;
return 0;
}
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 分享4款.NET开源、免费、实用的商城系统
· 全程不用写代码,我用AI程序员写了一个飞机大战
· MongoDB 8.0这个新功能碉堡了,比商业数据库还牛
· 白话解读 Dapr 1.15:你的「微服务管家」又秀新绝活了
· 上周热点回顾(2.24-3.2)