信息学奥赛一本通 1311:【例2.5】求逆序对

题目

题目描述】

给定一个序列a_1,a_2,...a_n,如果存在i < j并且a_i>a_j,那么我们称之为逆序对,求逆序对的数目。

【输入】

第一行为n,表示序列长度,接下来的n行,第i+1行表示序列中的第i个数。

【输出】

所有逆序对总数。

【输入样例】

4
3
2
3
2

【输出样例】

3

【提示】

n \le 10^5, A_i \le 10^5

代码

#include<iostream>
using namespace std;
const int N = 100111;
long long ans = 0;    //需要longlong,不然溢出
// 归并排序求逆序对

// 将a[first...mid]和a[mid+1...last]合并到r[first...last]中
void merge(int a[], int r[], int first, int mid, int last) {
	int i = first, j = mid + 1, k = first;
	while (i <= mid && j <= last) {
		if (a[i] > a[j]) r[k++] = a[j++], ans += mid - i + 1;
		else r[k++] = a[i++];
	}
	//拷贝剩余的数
	while (i <= mid) r[k++] = a[i++];
	while (j <= last) r[k++] = a[j++];
	//拷贝到原数组
	for (int i = first; i <= last; i++) a[i] = r[i];
}

// 归并排序
void mergesort(int a[], int r[], int first, int last) {
	if (first >= last) return;
	int mid = first + (last - first) / 2;
	mergesort(a, r, first, mid);
	mergesort(a, r, mid + 1, last);
	merge(a, r, first, mid, last);
}
int a[N], tmp[N];
int main() {
	int n;
	cin >> n;
	for (int i = 0; i < n; i++) cin >> a[i];
	mergesort(a, tmp, 0, n - 1);
	cout << ans << endl;
	return 0;
}

评测结果

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值