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. 学习 MySQL-DBA常用SQL汇总

    创建用户 GRANT USAGE ON *.* TO 'rp'@'%' IDENTIFIED BY 'rp' MAX_UPDATES_PER_HOUR MAX_CONNECTIONS_PER_HOUR ...

  2. 转载文章----初识Ildasm.exe——IL反编译的实用工具

    转载地址http://www.cnblogs.com/yangmingming/archive/2010/02/03/1662307.html Ildasm.exe 概要:(路径:C:\Program ...

  3. PlantUML的实例参考

    project: blog target: plant-uml-instances.md date: 2015-12-24 status: publish tags: - PlantUML - UML ...

  4. Amoeba for MySQL---分布式数据库Proxy解决方案

    Amoeba是什么? Amoeba(变形虫)项目,致力于MySQL的分布式数据库前端代理层,它主要在应用层访问MySQL的时候充当SQL路由功能,专注于分布式数据库代理层(Database Proxy ...

  5. 导入导出oracle数据库表的dmp文件

    1.先进入命令行,点击开始,输入cmd 2.导入的命令是:imp 用户名/密码@网络服务名 file=xxx.dmp full=y; 3.导出的命令是:exp 用户名/密码@网络服务名 file=xx ...

  6. OEM代工厂产品经理个人经历谈

    创业不是一件随随便便的事情! 到2007年时,我已经在上海.广州.东莞三地的工厂打工有十来年了.正是这个时间结点,我也即将做父亲了.打了很久的工后,就开始感到疲倦,做来做去,都是给老板做,也就在这时开 ...

  7. js实现页面跳转的几种方式

    第一种:    <script language="javascript" type="text/javascript">           wi ...

  8. Makefile目标,伪目标,头文件自动依赖

    目标 即我们最终要生成的文件,make默认生成第一个目标,注意 makefile中tab和空格不是一回事,规则使用tab缩进,编辑器不要设置诸如"将tab替换为空格之类的选项",目 ...

  9. flock — 轻便的咨询文件锁定

    bool flock  ( resource $handle  , int $operation  [, int &$wouldblock  ] ) handle  文件系统指针,是典型地由 ...

  10. html之div拖拽,html5拖拽

    html之div拖拽 http://www.w3school.com.cn/html5/html_5_draganddrop.asp