站外题求助
  • 板块题目总版
  • 楼主qqqqq111
  • 当前回复10
  • 已保存回复10
  • 发布时间2021/8/18 09:35
  • 上次更新2023/11/4 10:14:59
查看原帖
站外题求助
313137
qqqqq111楼主2021/8/18 09:35

1237:求排列的逆序数

【题目描述】 在Internet上的搜索引擎经常需要对信息进行比较,比如可以通过某个人对一些事物的排名来估计他(或她)对各种不同信息的兴趣,从而实现个性化的服务。

对于不同的排名结果可以用逆序来评价它们之间的差异。考虑1,2,…,n的排列i1,i2,…,in,如果其中存在j,k,满足j<k,且ij>ik,那么就称(ij,ik)是这个排列的一个逆序。

一个排列含有逆序的个数称为这个排列的逆序数。例如排列 263451 含有8个逆序(2,1),(6,3),(6,4),(6,5),(6,1),(3,1),(4,1),(5,1),因此该排列的逆序数就是8。显然,由1,2,…,n 构成的所有n!个排列中,最小的逆序数是0,对应的排列就是1,2,…,n;最大的逆序数是n(n−1)2,对应的排列就是n,(n−1),…,2,1。逆序数越大的排列与原始排列的差异度就越大。

现给定1,2,…,n的一个排列,求它的逆序数。

【输入】 第一行是一个整数n,表示该排列有n个数(n<=100000)。

第二行是n个不同的正整数,之间以空格隔开,表示该排列。

【输出】 输出该排列的逆序数。

【输入样例】 6 2 6 3 4 5 1 【输出样例】 8

#include<cstdio>

#define MAXN 1000000
#define	int long long

int n;
int a[MAXN];
int c[MAXN];
int ans;

void msort(int l, int r) {
	if(l==r)
		return ;
	int mid=l+(r-l)/2;
	msort(l, mid);
	msort(mid+1, r);
	
	int i=l, j=mid+1, k=l;
	while(i<=mid && j<=r) {
		if(a[i]<=a[j])
			c[k++]=a[i++];
		else
			c[k++]=a[j++], ans+=mid-i+1;
	}
	
	while(i<=mid)
		c[k++]=a[i++];
	while(j<=r)
		c[k++]=a[j++];
		
	for(int i=l; i<=r; i++)
		a[i]=c[i];
}


signed main() {
	scanf("%d", &n);
	for(int i=0; i<n; i++)
		scanf("%d", &a[i]);
	
	msort(0, n-1);
	
	printf("%d", ans);
	return 0;
}

rt,求调qwq

2021/8/18 09:35
加载中...