洛古 P1774 最接近神的人
置顶题解 P1908 【逆序对】 - zhylj 的博客 - 洛谷博客
链接里有过程图,但题解里的题大小顺序和本题是相反的,但问题不大
题目描述
破解了符文之语,小FF开启了通往地下的道路。当他走到最底层时,发现正前方有一扇巨石门,门上雕刻着一幅古代人进行某种活动的图案。而石门上方用古代文写着“神的殿堂”。小FF猜想里面应该就有王室的遗产了。但现在的问题是如何打开这扇门……
仔细研究后,他发现门上的图案大概是说:古代人认为只有智者才是最容易接近神明的。而最聪明的人往往通过一种仪式选拔出来。仪式大概是指,即将隐退的智者为他的候选人写下一串无序的数字,并让他们进行一种操作,即交换序列中相邻的两个元素。而用最少的交换次数使原序列变成不下降序列的人即是下一任智者。
小FF发现门上同样有着n个数字。于是他认为打开这扇门的秘诀就是找到让这个序列变成不下降序列所需要的最小次数。但小FF不会……只好又找到了你,并答应事成之后与你三七分……
输入格式
第一行为一个整数n,表示序列长度
第二行为n个整数,表示序列中每个元素。
输出格式
一个整数ans,即最少操作次数。
输入输出样例
输入 #1复制
4 2 8 0 3
输出 #1复制
3
说明/提示
对于30%的数据1≤n≤10^4。
对于100%的数据1≤n≤5*10^5;
-maxlongint≤A[i]≤maxlongint。
样例说明:开始序列为2 8 0 3,目标序列为0 2 3 8,可进行三次操作的目标序列:
1.Swap (8,0):2 0 8 3
2.Swap (2,0):0 2 8 3
3.Swap (8,3):0 2 3 8
附件:
#include<bits/stdc++.h>
using namespace std;
const int N = 5e5 + 2;
int c[N],a[N],b[N],n;
#define ll long long
ll ans;
bool cmp(int x,int y)
{
return a[x] < a[y];
}
void add(int x)
{
while(x <= n){
++c[x],x += (x & -x);
}
return;
}
int ask(int x)
{
int s = 0;
while(x){
s += c[x];
x -= (x & -x);
}
return s;
}
int main()
{
scanf("%d",&n);
for(int i = 1;i <= n;i ++)
{
scanf("%d",&a[i]);
b[i] = i;
}
stable_sort(b + 1,b + n + 1,cmp);//带有stable的函数可保证相等元素的原本相对次序在排序后保持不变
// for(int i = 1;i <= n;++ i)
//a[b[i]] = i;
/*
a数组值改为相对大小
*/
for(int i = 1;i <= n;i ++)
cout << a[i] << endl;
for(int i = n;i >= 1;i --)
{
ans += ask(a[i] - 1);
/*
从大到小枚举
排在后面的先进去,如果不是逆序对,那么后面进来的都在这个数对应的数的后面(树状数组图里),即相对大小比该数大,求他的前缀和就是求有几个逆序对
*/
cout << "ans:" << ans <<endl;
add(a[i]);
}
printf("%lld\n",ans);
return 0;
}