【BZOJ】【3295】【CQOI2011】动态逆序对

时间:2025-03-31 23:04:49

树套树


  Orz zyf神犇

  时光倒流……逆序处理,将删点改为加点,动态维护序列。

  由于是动态,要加点,所以用树状数组;同时又需要求序列中求比当前元素大/小的元素个数,所以要用平衡树。

  所以方法就是在树状数组的每个节点上维护一棵这个节点表示的区间的平衡树。

  为什么这样做是对的呢?因为求<k的元素个数这类操作满足区间加法,所以可以把多棵平衡树上的结果(一棵平衡树表示一个区间)加起来,就是整个区间的结果。

(我一开始想成带修改的区间第K大那种做法了……就是树状数组套权值线段树……sigh)

WA:ans[m+1]需要初始化,样例是因为只剩一个数所以逆序对数为0,但是其他数据明显不会是这样……

 /**************************************************************
Problem: 3295
User: Tunix
Language: C++
Result: Accepted
Time:8432 ms
Memory:122372 kb
****************************************************************/ //BZOJ 3295
#include<cmath>
#include<vector>
#include<cstdio>
#include<cstring>
#include<cstdlib>
#include<iostream>
#include<algorithm>
#define rep(i,n) for(int i=0;i<n;++i)
#define F(i,j,n) for(int i=j;i<=n;++i)
#define D(i,j,n) for(int i=j;i>=n;--i)
#define pb push_back
#define CC(a,b) memset(a,b,sizeof(a))
using namespace std;
int getint(){
int v=,sign=; char ch=getchar();
while(!isdigit(ch)) {if(ch=='-') sign=-; ch=getchar();}
while(isdigit(ch)) {v=v*+ch-''; ch=getchar();}
return v*sign;
}
const int N=,M=,INF=~0u>>;
const double eps=1e-;
typedef long long LL;
/*******************template********************/
int n,m,tot,l[M],r[M],s[M],rnd[M],w[M],v[M];
#define L l[x]
#define R r[x]
inline void Push_up(int x){
s[x]=s[L]+s[R]+w[x];
}
inline void zig(int &x){
int t=L; L=r[t]; r[t]=x; s[t]=s[x]; Push_up(x); x=t;
}
inline void zag(int &x){
int t=R; R=l[t]; l[t]=x; s[t]=s[x]; Push_up(x); x=t;
}
void ins(int &x,int num){
if (!x){
x=++tot; v[x]=num; s[x]=w[x]=; L=R=; rnd[x]=rand(); return;
}
s[x]++;
if (v[x]==num) w[x]++;
else if(num<v[x]){
ins(L,num); if(rnd[L]<rnd[x]) zig(x);
}else{
ins(R,num); if(rnd[R]<rnd[x]) zag(x);
}
}
int rank(int x,int num){//比x小的数有多少个
if (!x) return ;
if (v[x]==num) return s[L];
else if(num<v[x]) return rank(L,num);
else return s[L]+w[x]+rank(R,num);
}
#undef L
#undef R
/**********************Treap********************/
int rt[N];
int ss[N];
void update(int x,int y){
for(int i=x;i<=n;i+=i&-i) ins(rt[i],y);
}
LL getbig(int x,int val){
int t=;
for(int i=x;i;i-=i&-i)
t+=s[rt[i]]-rank(rt[i],val);
return t;
}
LL getsml(int x,int y,int val){
int t1=,t2=;
for(int i=x;i;i-=i&-i)
t1+=rank(rt[i],val);
for(int j=y;j;j-=j&-j)
t2+=rank(rt[j],val);
return t2-t1;
}
void add(int x){
for(int i=x;i<=n;i+=i&-i) ss[i]++;
}
LL sum(int x){
int ans=;
for(int i=x;i;i-=i&-i) ans+=ss[i];
return ans;
}
/*********************Fenwick*******************/
struct data{
int v,pos;
bool operator < (const data&b) const{
return v<b.v;
}
}a[N],b[N],d[N];
LL ans[N];
int main(){
#ifndef ONLINE_JUDGE
freopen("input.txt","r",stdin);
// freopen("output.txt","w",stdout);
#endif
n=getint(); m=getint();
F(i,,n) a[i].v=getint(),a[i].pos=i,b[i]=a[i];
sort(b+,b+n+);
F(i,,m){
d[i].v=getint();
d[i].pos=b[d[i].v].pos;
}
F(i,,n) b[i]=a[i];
F(i,,m) b[d[i].pos].v=;
ans[m+]=; F(i,,n) if (b[i].v){
update(i,b[i].v);
ans[m+]+=sum(n)-sum(b[i].v);
add(b[i].v);
} D(i,m,){
ans[i]=ans[i+]+getbig(d[i].pos-,d[i].v)+getsml(d[i].pos,n,d[i].v);
update(d[i].pos,d[i].v);
}
F(i,,m) printf("%lld\n",ans[i]);
return ;
}