• BZOJ3295 [Cqoi2011]动态逆序对 分治 树状数组


    原文链接http://www.cnblogs.com/zhouzhendong/p/8678185.html

    题目传送门 - BZOJ3295

    题意

      对于序列$A$,它的逆序对数定义为满足$i<j$,且$A_i>A_j$的数对$(i,j)$的个数。给$1$到$n$的一个排列,按照某种顺序依次删除$m$个元素,你的任务是在每次删除一个元素之前统计整个序列的逆序对数。

    题解

      我们首先把原题目转化成依次加入数字求总逆序对个数。

      假设某一个数字被加入的时间为$t$,他的位置为$id$,它的值为$v$。

      则存在两种情况,使得$i$能更新$j$。

      $Situation 1:$

      $t_i<t_j,id_i<id_j,v_i>v_j$

      $Situation 2:$

      $t_i<t_j,id_i>id_j,v_i<v_j$

      于是机智的你是不是发现CDQ分治一下就秒掉了???

      (其实这题如果用带修改的主席树或者树套树貌似脑子都不用动…………)

    代码

    #include <bits/stdc++.h>
    using namespace std;
    typedef long long LL;
    const int N=100005;
    struct Node{
    	int id,v,t,res;
    }a[N],b[N];
    int n,m,pos[N],tree[N];
    LL res[N];
    int lowbit(int x){
    	return x&-x;
    }
    void add(int x,int y){
    	for (;x<=n;x+=lowbit(x))
    		tree[x]+=y;
    }
    int sum(int x){
    	int ans=0;
    	for (;x>0;x-=lowbit(x))
    		ans+=tree[x];
    	return ans;
    }
    void CDQ(int L,int R){
    	if (L==R)
    		return;
    	int mid=(L+R)>>1;
    	for (int i=L,l=L,r=mid+1;i<=R;i++)
    		if (a[i].t<=mid)
    			b[l++]=a[i];
    		else
    			b[r++]=a[i];
    	for (int i=L;i<=R;i++)
    		a[i]=b[i];
    	int j=L;
    	for (int i=mid+1;i<=R;i++){
    		while (j<=mid&&a[j].id<a[i].id)
    			add(n+1-a[j].v,1),j++;
    		a[i].res+=sum(n+1-a[i].v);
    	}
    	for (int i=L;i<j;i++)
    		add(n+1-a[i].v,-1);
    	j=mid;
    	for (int i=R;i>mid;i--){
    		while (j>=L&&a[j].id>a[i].id)
    			add(a[j].v,1),j--;
    		a[i].res+=sum(a[i].v);
    	}
    	for (int i=mid;i>j;i--)
    		add(a[i].v,-1);
    	CDQ(L,mid),CDQ(mid+1,R);
    }
    int main(){
    	scanf("%d%d",&n,&m);
    	for (int i=1;i<=n;i++){
    		scanf("%d",&a[i].v);
    		a[i].id=pos[a[i].v]=i;
    		a[i].t=a[i].res=0;
    	}
    	for (int i=1,x;i<=m;i++){
    		scanf("%d",&x);
    		a[pos[x]].t=n-i+1;
    	}
    	for (int i=1,t=n-m;i<=n;i++)
    		if (!a[i].t)
    			a[i].t=t--;
    	memset(tree,0,sizeof tree);
    	CDQ(1,n);
    	memset(res,0,sizeof res);
    	for (int i=1;i<=n;i++)
    		res[a[i].t]+=a[i].res;
    	for (int i=2;i<=n;i++)
    		res[i]+=res[i-1];
    	for (int i=n;i>n-m;i--)
    		printf("%lld
    ",res[i]);
    	return 0;
    }
    

      

  • 相关阅读:
    Python中max()内置函数使用(list)
    linux命令总结mpstat命令
    linux命令总结vmstat命令
    linux命令总结free命令
    linux命令总结top命令
    linux命令总结之echo命令
    Nagios服务器端配置文件详解
    python之OS模块详解
    真正的inotify+rsync实时同步 彻底告别同步慢
    linux BASH shell下设置字体及背景颜色
  • 原文地址:https://www.cnblogs.com/zhouzhendong/p/BZOJ3295.html
Copyright © 2020-2023  润新知