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. 不用画线 设置UITableView的全屏分隔线

    如图  添加如下代码 sTableView.separatorInset = UIEdgeInsetsZero; sTableView.layoutMargins = UIEdgeInsetsZero ...

  2. Sql server profiler抓出的语句不可信

      Sql profiler抓出的语句不可信      

  3. flashdevelop生成swc库

    flashdevelop没有直接支持生成swc的工程,但flashdevelop生成swc也比较方便,不用任何插件. swc库是由 flexsdk的compc.exe生成的,其实我们通过这个命令行也可 ...

  4. 回到过去美好的时光——源代码版本管理Always Use source code Control

    Don't Repeat Yourself Don't Repeat Yourself,这是程序员修炼之道中的经典名言.源代码版本管理对程序员来说是非常重要的工作.因为它让你能够回到过去,而不用做重复 ...

  5. 最新中国菜刀caidao-20160620下载和说明

    0x00 中国制造, 黑客之刀 中国菜刀是中国安全圈内使用非常广泛的一款Webshell管理工具,此次老兵大牛进行了更新,界面更加优美,功能更加丰富,有图有真相! 0x01 验证 此次更新还自带了官网 ...

  6. [gist]在浏览器里免查看源代码格式化var_dump输出

    Gist Link /** * 格式化var_dump输出... * 我勒个去..早怎么没想到..就加了个pre啊,, */ function var_dump_html($var){ echo &q ...

  7. mapreduce流程中的几个关键点

    MapReduce中数据流动    (1)最简单的过程:  map - reduce    (2)定制了partitioner以将map的结果送往指定reducer的过程: map - partiti ...

  8. mysql自动备份维护shell脚本 (copy)

    #!/bin/bash #Mysql 自动备份 压缩并上传到 指定ftp #设想每天凌晨3点备份mysql #编辑crontab配置文件 # * * * backupmysql.sh #压缩并以&qu ...

  9. my_atoi()

    void my_atoi(const char* s){ int i=0,res=0; if(*s<='9' && *s>='0'){ //如果输入的一个字符是数字 for ...

  10. ps, top, pstree

    ps 查看当前终端所启动的进程, 不加选项只查看当前终端的进程 PID TTY TIME CMD 2398 pts/1 00:00:00 bash 3625 pts/1 00:00:00 ps #PI ...