【BZOJ2141】排队(CDQ分治)
【BZOJ2141】排队(CDQ分治)
题面
题解
大部分树套树/主席树这类题目都可以用整体二分/CDQ分治来做。
这题考虑一下,在不考虑修改的情况下
贡献是如何产生的?
我们发现是个二位偏序问题(或者说是一个逆序对修改版本)
现在有了一个修改,那么产生贡献的前提额外增加一个:时间。
既然变成了一个三位偏序问题
考虑\(CDQ\)分治
按照时间分治,块内按照\(x\)排序,考虑左侧对右侧的贡献:
维护当前数字(离散后)的一个值域树状数组
因为贡献有当前点作为左端点和右端点的两部分
所以,按照\(x\)正着加入树状数组一次,反着加入树状数组一次。
就求一下在当前时间之前,产生贡献的值就行了。
但是交换操作很不好办。
我们可以把一个交换操作改成两个删除操作和两个插入操作。
这样就可以交换的问题。
一个额外要注意的问题:排序的时候,如果\(x\)相同,一定还要按照修改的值排序,因为小的值同样可以更新大的值,否则会错。
#include<iostream>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<cmath>
#include<algorithm>
#include<set>
#include<map>
#include<vector>
#include<queue>
using namespace std;
#define ll long long
#define RG register
#define MAX 22222
#define lb(x) (x&(-x))
inline int read()
{
RG int x=0,t=1;RG char ch=getchar();
while((ch<'0'||ch>'9')&&ch!='-')ch=getchar();
if(ch=='-')t=-1,ch=getchar();
while(ch<='9'&&ch>='0')x=x*10+ch-48,ch=getchar();
return x*t;
}
struct Operator{int t,x,y,w,id;}q[MAX<<2],tmp[MAX<<2];
bool operator<(Operator a,Operator b){if(a.x!=b.x)return a.x<b.x;return a.y<b.y;}
int tot,len,S[MAX],a[MAX],n,m,tim;
int c[MAX],ans[MAX];
void add(int x,int w){while(x<=n)c[x]+=w,x+=lb(x);return;}
int getsum(int x){int ret=0;while(x)ret+=c[x],x-=lb(x);return ret;}
void CDQ(int l,int r)
{
if(l==r)return;
int mid=(l+r)>>1;
for(int i=l;i<=r;++i)
if(q[i].t<=mid)add(q[i].y,q[i].w);
else ans[q[i].id]+=q[i].w*(getsum(n)-getsum(q[i].y));
for(int i=l;i<=r;++i)
if(q[i].t<=mid)add(q[i].y,-q[i].w);
for(int i=r;i>=l;--i)
if(q[i].t<=mid)add(q[i].y,q[i].w);
else ans[q[i].id]+=q[i].w*getsum(q[i].y-1);
for(int i=l;i<=r;++i)if(q[i].t<=mid)add(q[i].y,-q[i].w);
int t1=l-1,t2=mid;
for(int i=l;i<=r;++i)
if(q[i].t<=mid)tmp[++t1]=q[i];
else tmp[++t2]=q[i];
for(int i=l;i<=r;++i)q[i]=tmp[i];
CDQ(l,mid);CDQ(mid+1,r);
}
int main()
{
n=read();
for(int i=1;i<=n;++i)a[i]=S[i]=read();
sort(&S[1],&S[n+1]);len=unique(&S[1],&S[n+1])-S-1;
for(int i=1;i<=n;++i)a[i]=lower_bound(&S[1],&S[len+1],a[i])-S;
for(int i=1;i<=n;++i)q[++tot]=(Operator){++tim,i,a[i],1,0};
n=len;m=read();
for(int i=1;i<=m;++i)
{
int x=read(),y=read();
q[++tot]=(Operator){++tim,x,a[y],+1,i};
q[++tot]=(Operator){++tim,y,a[x],+1,i};
q[++tot]=(Operator){++tim,x,a[x],-1,i};
q[++tot]=(Operator){++tim,y,a[y],-1,i};
swap(a[x],a[y]);
}
sort(&q[1],&q[tot+1]);
CDQ(1,tim);
printf("%d\n",ans[0]);
for(int i=1;i<=m;++i)printf("%d\n",ans[i]+=ans[i-1]);
return 0;
}
【BZOJ2141】排队(CDQ分治)的更多相关文章
- BZOJ 2141: 排队 [CDQ分治]
题意: 交换序列中两个元素,求逆序对 做分块做到这道题...一看不是三维偏序嘛.... 作为不会树套树的蒟蒻就写CDQ分治吧.... 对时间分治...x排序...y树状数组... 交换拆成两个插入两个 ...
- [国家集训队]排队 [cdq分治]
题面 洛谷 和动态逆序对那道题没有什么区别 把一个交换换成两个删除和两个插入 #include <cstdio> #include <cstdlib> #include < ...
- bzoj 2141 : 排队 (cdq分治+bit)
链接: https://www.lydsy.com/JudgeOnline/problem.php?id=2141 思路: 其实就是求动态逆序对...cdq降维,用树状数组前后求两遍逆序对就好了 切水 ...
- BZOJ 2141 排队 (CDQ分治)
[BZOJ2141]排队 这道题和动态逆序对比较像(BZOJ-3295 没做过的同学建议先做这题),只是删除操作变成了交换.解法:交换操作可以变成删除加插入操作,那么这题就变成了 (时间,位置,值)的 ...
- 【教程】简易CDQ分治教程&学习笔记
前言 辣鸡蒟蒻__stdcall终于会CDQ分治啦! CDQ分治是我们处理各类问题的重要武器.它的优势在于可以顶替复杂的高级数据结构,而且常数比较小:缺点在于必须离线操作. CDQ分治的基 ...
- BZOJ 2683 简单题 ——CDQ分治
[题目分析] 感觉CDQ分治和整体二分有着很本质的区别. 为什么还有许多人把他们放在一起,也许是因为代码很像吧. CDQ分治最重要的是加入了时间对答案的影响,x,y,t三个条件. 排序解决了x ,分治 ...
- HDU5618 & CDQ分治
Description: 三维数点 Solution: 第一道cdq分治...感觉还是很显然的虽然题目不能再傻逼了... Code: /*=============================== ...
- 初识CDQ分治
[BZOJ 1176:单点修改,查询子矩阵和]: 1176: [Balkan2007]Mokia Time Limit: 30 Sec Memory Limit: 162 MBSubmit: 200 ...
- HDU5322 Hope(DP + CDQ分治 + NTT)
题目 Source http://acm.hdu.edu.cn/showproblem.php?pid=5322 Description Hope is a good thing, which can ...
随机推荐
- 解决Extjs有IE下z-index属性的问题
在用Extjs时,有时候,在Google浏览器上面没有任何问题,但是相同的页面在IE下面就会有问题,直接报错,点击中断,进行后可以看到如下的信息: Google里面没这个问题,加一句代码就能解决在窗体 ...
- 关于ExecuteNonQuery执行的返回值(SQL语句、存储过程)
因为msdn中说返回受影响的行数: Executes a Transact-SQL statement against the connection and returns the number of ...
- scrapy 爬取知乎问题、答案 ,并异步写入数据库(mysql)
python版本 python2.7 爬取知乎流程: 一 .分析 在访问知乎首页的时候(https://www.zhihu.com),在没有登录的情况下,会进行重定向到(https://www. ...
- java excel导出(表头合并,多行表头)
@RequestMapping(value="orderExcelList2") public void orderExcelList2forJava(Order order,Ht ...
- Java EE JSP内置对象及表达式语言
一.JSP内置对象 JSP根据Servlet API规范提供了一些内置对象,开发者不用事先声明就可使用标准变量来访问这些对象. JSP提供了9种内置对象: (一).request 简述: JSP编程中 ...
- Tensorflow基本开发架构
Tensorflow基本开发架构 先说句题外话, 这段时间一直研究爬虫技术,主要目的是为将来爬取训练数据做准备,同时学习python编程.这一研究才发现,python的开发资源实在是太丰富了,所有你能 ...
- MAC下Android的Eclipse开发环境搭建
原文链接:https://www.cnblogs.com/macro-cheng/archive/2011/09/30/android-001.html 一.Eclipse的下载 到网站:http:/ ...
- 3.0 zookeeper的集群介绍、搭建、环境、安装
zookeeper是本身是一种分布式协调服务(英文意思动物园园长因为Hadoop就是一个动物园,storm.hadoop.kafkaka.hbaser都是基于zookeeper开发的) 原理:Zook ...
- JUnit initializationError错误
一.JUnit Test 测试 initializationError错误 MyMaincom.test.sunc.MyMaininitializationError(com.test.sunc.My ...
- .NET 4.0 Tuple 元组
Tuple是.NET 4.0的新特性,主要功能是动态返回数据结构,也可以用做临时数据结构. 原来做一些功能时需要一个方法返回几个值,有两种方法: 1. 非常难看.难用的OUT参数: 2. 新写一个实体 ...