3295: [Cqoi2011]动态逆序对

Time Limit: 10 Sec  Memory Limit: 128 MB
Submit: 3865  Solved: 1298
[Submit][Status][Discuss]

Description

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

Input

输入第一行包含两个整数nm,即初始元素的个数和删除的元素个数。以下n行每行包含一个1到n之间的正整数,即初始排列。以下m行每行一个正整数,依次为每次删除的元素。
 

Output

 
输出包含m行,依次为删除每个元素之前,逆序对的个数。

Sample Input

5 4
1
5
3
4
2
5
1
4
2

Sample Output

5
2
2
1

样例解释
(1,5,3,4,2)(1,3,4,2)(3,4,2)(3,2)(3)。

HINT

N<=100000 M<=50000

Source

树状数组套线段树

删除某个数,只要统计它之前还存在的比它大的数的个数,和之后还存在的比它小的数的个数

#include<cstdio>
#include<cstring>
#include<algorithm>
#define ll long long
#define R register
using namespace std;
int read(){
R int x=;bool f=;
R char ch=getchar();
while(ch<''||ch>''){if(ch=='-')f=;ch=getchar();}
while(ch>=''&&ch<=''){x=(x<<)+(x<<)+ch-'';ch=getchar();}
return f?x:-x;
}
const int N=1e5+;
const int M=N*;
int A[],B[];
int n,m,sz,num[N],pos[N],a1[N],a2[N],root[N],c[N];
int ls[M],rs[M],sum[M];
ll ans;
inline int lowbit(int x){
return x&-x;
}
void updata(int p,int v){
for(int i=p;i<=n;i+=lowbit(i)) c[i]+=v;
}
int query(int p){
int res=;
for(int i=p;i;i-=lowbit(i)) res+=c[i];
return res;
}
void update(int &y,int l,int r,int x){
if(!y) y=++sz;
sum[y]++;
if(l==r)return;
int mid=(l+r)>>;
if(x<=mid) update(ls[y],l,mid,x);
else update(rs[y],mid+,r,x);
}
int askmore(int x,int y,int num){
A[]=B[]=;int tmp=;x--;
for(int i=x;i;i-=lowbit(i)) A[++A[]]=root[i];
for(int i=y;i;i-=lowbit(i)) B[++B[]]=root[i];
int l=,r=n;
while(l!=r){
int mid=l+r>>;
if(num<=mid){
for(int i=;i<=A[];i++) tmp-=sum[rs[A[i]]];
for(int i=;i<=B[];i++) tmp+=sum[rs[B[i]]];
for(int i=;i<=A[];i++) A[i]=ls[A[i]];
for(int i=;i<=B[];i++) B[i]=ls[B[i]];
r=mid;
}
else{
for(int i=;i<=A[];i++) A[i]=rs[A[i]];
for(int i=;i<=B[];i++) B[i]=rs[B[i]];
l=mid+;
}
}
return tmp;
}
int askless(int x,int y,int num){
A[]=B[]=;int tmp=;x--;
for(int i=x;i;i-=lowbit(i)) A[++A[]]=root[i];
for(int i=y;i;i-=lowbit(i)) B[++B[]]=root[i];
int l=,r=n;
while(l!=r){
int mid=l+r>>;
if(num>mid){
for(int i=;i<=A[];i++) tmp-=sum[ls[A[i]]];
for(int i=;i<=B[];i++) tmp+=sum[ls[B[i]]];
for(int i=;i<=A[];i++) A[i]=rs[A[i]];
for(int i=;i<=B[];i++) B[i]=rs[B[i]];
l=mid+;
}
else{
for(int i=;i<=A[];i++) A[i]=ls[A[i]];
for(int i=;i<=B[];i++) B[i]=ls[B[i]];
r=mid;
}
}
return tmp;
}
int main(){
n=read();m=read();
for(int i=;i<=n;i++){
num[i]=read();pos[num[i]]=i;
a1[i]=query(n)-query(num[i]);
ans+=a1[i];
updata(num[i],);
}
memset(c,,sizeof c);
for(int i=n;i;i--){
a2[i]=query(num[i]-);
updata(num[i],);
}
for(int i=,x;i<=m;i++){
printf("%lld\n",ans);
x=read();x=pos[x];
ans-=(a1[x]+a2[x]-askmore(,x-,num[x])-askless(x+,n,num[x]));
for(int j=x;j<=n;j+=lowbit(j)) update(root[j],,n,num[x]);
}
return ;
}

BZOJ 3295: [Cqoi2011]动态逆序对的更多相关文章

  1. Bzoj 3295: [Cqoi2011]动态逆序对 分块,树状数组,逆序对

    3295: [Cqoi2011]动态逆序对 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 2886  Solved: 924[Submit][Stat ...

  2. bzoj 3295 [Cqoi2011]动态逆序对(cdq分治,BIT)

    [题目链接] http://www.lydsy.com/JudgeOnline/problem.php?id=3295 [题意] n个元素依次删除m个元素,求删除元素之前序列有多少个逆序对. [思路] ...

  3. 【刷题】BZOJ 3295 [Cqoi2011]动态逆序对

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

  4. bzoj 3295: [Cqoi2011]动态逆序对(树套树 or CDQ分治)

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

  5. BZOJ 3295: [Cqoi2011]动态逆序对 [CDQ分治]

    RT 传送门 首先可以看成倒着插入,求逆序对数 每个数分配时间(注意每个数都要一个时间)$t$,$x$位置,$y$数值 $CDQ(l,r)$时归并排序$x$ 然后用$[l,mid]$的加入更新$[mi ...

  6. BZOJ 3295 [CQOI2011]动态逆序对 (三维偏序CDQ+树状数组)

    题目大意: 题面传送门 还是一道三维偏序题 每次操作都可以看成这样一个三元组 $<x,w,t>$ ,操作的位置,权值,修改时间 一开始的序列看成n次插入操作 我们先求出不删除时的逆序对总数 ...

  7. BZOJ 3295 [Cqoi2011]动态逆序对 ——CDQ分治

    时间.位置.数字为三个属性. 排序时间,CDQ位置,树状数组处理数字即可. #include <cstdio> #include <cstring> #include < ...

  8. 【BZOJ 3295】动态逆序对 - 分块+树状数组

    题目描述 给定一个1~n的序列,然后m次删除元素,每次删除之前询问逆序对的个数. 分析:分块+树状数组 (PS:本题的CDQ分治解法见下一篇) 首先将序列分成T块,每一块开一个树状数组,并且先把最初的 ...

  9. 【Bzoj 3295】 动态逆序对(树套树|CDQ分治)

    [题意] 每次删除一个数,然后问删除前逆序对数. [分析] 没有AC不开心.. 我的树状数组套字母树,应该是爆空间的,空间复杂度O(nlogn^2)啊..哭.. 然后就没有然后了,别人家的树套树是树状 ...

随机推荐

  1. 一起来学习Android自定义控件1

    概述 Android已经为我们提供了大量的View供我们使用,但是可能有时候这些组件不能满足我们的需求,这时候就需要自定义控件了.自定义控件对于初学者总是感觉是一种复杂的技术.因为里面涉及到的知识点会 ...

  2. IOS NSTimer和CADisplayLink的用法

    IOS--NSTimer和CADisplayLink的用法 NSTimer初始化器接受调用方法逻辑之间的间隔作为它的其中一个参数,预设一秒执行30次.CADisplayLink默认每秒运行60次,通过 ...

  3. 【原】ios的hitTest方法以及不规则区域内触摸事件处理方法

    概述 在正常的使用场景中,我们处理了比较多的矩形区域内触摸事件,比如UIButton.UIControl.一般来说,这些控件的图形以及触摸区域都是矩形或者圆角矩形的.但是在一些特殊应用场景中我们有时不 ...

  4. 基于ruby的watir自动化测试 笔记二

    基于ruby的watir自动化测试 笔记一的补充版,新增加了些特殊的控件捕获方法.还在更新中.... attribute_value 获取当前控件的属性 Value = ie.link(:id=> ...

  5. symfony2 twig模板引擎

    1.基本语法 Says something:{{    }} Does something:{%  %} Comment something:{#    #} {% extends "App ...

  6. CSS Float 以及相关布局模式

    float 取值 属性 值 描述   left 向左浮动   right 向右浮动   none 默认值   inherit 继承 看一个栗子 红色线框代表父元素 脱离文档流,其实也没有完全脱离,会被 ...

  7. android 基本布局(RelativeLayout、TableLayout等)使用方法及各种属性

        本文介绍 Android 界面开发中最基本的四种布局LinearLayout.RelativeLayout.FrameLayout.TableLayout 的使用方法及这四种布局中常用的属性. ...

  8. Windows下好用到必须开机自启的小工具

    折腾过linux,黑苹果,最后还是回到了盖茨大叔的windows.得出的结论是,日常使用的话,折腾Linux还不如把精力去拿去折腾windows.分享下折腾的成果,介绍下一些很不错的小工具.     ...

  9. php 升级到 5.3+ 后出现的一些错误,如 ereg(); ereg_replace(); 函数报错

    在php5.3环境下运行,常常会出现 Deprecated: Function ereg() is deprecated in...和Deprecated: Function ereg_replace ...

  10. C++/CLI——读书笔记《Visual C++/CLI从入门到精通》 第Ⅱ部分

    =================================版权声明================================= 版权声明:本文为博主原创文章 未经许可不得转载  请通过右 ...