P4390 [BOI2007]Mokia 摩基亚

题目描述

摩尔瓦多的移动电话公司摩基亚(\(Mokia\))设计出了一种新的用户定位系统。和其他的定位系统一样,它能够迅速回答任何形如“用户\(C\)的位置在哪?”的问题,精确到毫米。但其真正高科技之处在于,它能够回答形如“给定区域内有多少名用户?”的问题。

在定位系统中,世界被认为是一个\(W\times W\)的正方形区域,由\(1\times1\)的方格组成。每个方格都有一个坐标\((x,y)\),\(1\le x,y\le W\)。坐标的编号从\(1\)开始。对于一个\(4\times4\)的正方形,就有\(1\le x\le 4,1\le y\le 4\)(如图):

请帮助\(Mokia\)公司编写一个程序来计算在某个矩形区域内有多少名用户。

输入输出格式

输入格式:

有三种命令,意义如下:

命令 参数 意义

  • 0 W 初始化一个全零矩阵。本命令仅开始时出现一次。
  • 1 x y A 向方格\((x,y)\)中添加\(A\)个用户。\(A\)是正整数。
  • 2 X1 Y1 X2 Y2 查询\(X_1\le x\le X_2,Y_1\le y\le Y_2\)所规定的矩形中的用户数量
  • 3 无参数 结束程序。本命令仅结束时出现一次。

输出格式:

对所有命令\(2\),输出一个一行整数,即当前询问矩形内的用户数量。

说明

对于所有数据:

\(1\le W\le 2000000\)

\(1\le X_1\le X_2\le W\)

\(1\le Y_1\le Y_2\le W\)

\(1\le x,y\le W\)

\(0<A\le 10000\)

命令\(1\)不超过\(160000\)个。

命令\(2\)不超过\(10000\)个。


之前做了个园丁的题目,是所有询问在修改之后的。

于是直接把四维偏序的一维搞成区间操作\(O(n\log^2n)\)水过去了。

然后这个题就萎掉了,想了一会儿还是三个\(\log\),于是打算拿树套树水过去。

结果套的线段树\(MLE\)了,无语...

然后看了看正解,发现直接对矩形容斥就可以了。

就是把一个矩形\((a,b),(c,d)\)拆成\((1,1),(a-1,b-1)\)、\((1,1),(a-1,d)\)、\((1,1),(c,b-1)\)、\((1,1),(c,d)\)四个做。

然后就又成了三维偏序...


Code:

#include <cstdio>
#include <algorithm>
const int N=2e5+10;
int n,m,k,op,ans[N],s[N<<2],dx[N<<2],cntx,dy[N<<2],cnty;
void add(int x,int d){while(x<=cnty)s[x]+=d,x+=x&-x;}
int ask(int x){int sum=0;while(x)sum+=s[x],x-=x&-x;return sum;}
struct node
{
int a,b,c,op;
bool friend operator <(node n1,node n2)
{
return n1.a==n2.a?n1.op<n2.op:n1.a<n2.a;
}
}q[N],qs[N];
void CDQ(int l,int r)
{
if(l==r) return;
int mid=l+r>>1;
CDQ(l,mid),CDQ(mid+1,r);
int lp=l,rp=mid+1,loc=l-1;
while(lp<=mid&&rp<=r)
{
if(q[lp]<q[rp])
{
if(!q[lp].op) add(q[lp].b,q[lp].c);
qs[++loc]=q[lp++];
}
else
{
if(q[rp].op) ans[q[rp].op]+=q[rp].c*ask(q[rp].b);
qs[++loc]=q[rp++];
}
}
while(rp<=r)
{
if(q[rp].op) ans[q[rp].op]+=q[rp].c*ask(q[rp].b);
qs[++loc]=q[rp++];
}
for(int i=l;i<lp;i++) if(!q[i].op) add(q[i].b,-q[i].c);
while(lp<=mid) qs[++loc]=q[lp++];
for(int i=l;i<=r;i++) q[i]=qs[i];
}
int main()
{
scanf("%d%d",&n,&n);
scanf("%d",&op);int a,b,c,d;
while(op!=3)
{
if(op==1)
{
++m;
scanf("%d%d%d",&q[m].a,&q[m].b,&q[m].c);
dx[++cntx]=q[m].a,dy[++cnty]=q[m].b;
}
else
{
scanf("%d%d%d%d",&a,&b,&c,&d);
dx[++cntx]=a-1,dy[++cnty]=b-1,dx[++cntx]=c,dy[++cnty]=d;
++k;
q[++m]={a-1,b-1,1,k};
q[++m]={a-1,d,-1,k};
q[++m]={c,b-1,-1,k};
q[++m]={c,d,1,k};
}
scanf("%d",&op);
}
std::sort(dx+1,dx+1+cntx);
std::sort(dy+1,dy+1+cnty);
cntx=std::unique(dx+1,dx+1+cntx)-dx-1;
cnty=std::unique(dy+1,dy+1+cnty)-dy-1;
for(int i=1;i<=m;i++)
{
q[i].a=std::lower_bound(dx+1,dx+1+cntx,q[i].a)-dx;
q[i].b=std::lower_bound(dy+1,dy+1+cnty,q[i].b)-dy;
}
CDQ(1,m);
for(int i=1;i<=k;i++) printf("%d\n",ans[i]);
return 0;
}

2018.11.27

洛谷 P4390 [BOI2007]Mokia 摩基亚 解题报告的更多相关文章

  1. [洛谷P4390][BOI2007]Mokia 摩基亚

    题目大意: 维护一个W*W的矩阵,每次操作可以增加某格子的权值,或询问某子矩阵的总权值. 题解:CDQ分治,把询问拆成四个小矩形 卡点:无 C++ Code: #include <cstdio& ...

  2. P4390 [BOI2007]Mokia 摩基亚 (CDQ解决三维偏序问题)

    题目描述 摩尔瓦多的移动电话公司摩基亚(Mokia)设计出了一种新的用户定位系统.和其他的定位系统一样,它能够迅速回答任何形如"用户C的位置在哪?"的问题,精确到毫米.但其真正高科 ...

  3. Luogu P4390 [BOI2007]Mokia 摩基亚 | CDQ分治

    题目链接 $CDQ$分治. 考虑此时在区间$[l,r]$中,要计算$[l,mid]$中的操作对$[mid+1,r]$中的询问的影响. 计算时,排序加上树状数组即可. 然后再递归处理$[l,mid]$和 ...

  4. P4390 [BOI2007]Mokia 摩基亚(cdq分治)

    一样是cdq的板子 照着园丁的烦恼就好了 代码 #include <cstdio> #include <cstring> #include <algorithm> ...

  5. P4390 [BOI2007]Mokia 摩基亚

    传送门 对于一个询问 $(xa,ya),(xb,yb)$,拆成 $4$ 个询问并容斥一下 具体就是把询问变成求小于等于 $xb,yb$ 的点数,减去小于等于 $xa-1,yb$ 和小于等于 $xb,y ...

  6. 【BZOJ1176】[BOI2007]Mokia 摩基亚

    [BZOJ1176][BOI2007]Mokia 摩基亚 题面 bzoj 洛谷 题解 显然的\(CDQ\)\(/\)树套树题 然而根本不想写树套树,那就用\(CDQ\)吧... 考虑到点\((x1,y ...

  7. [BOI2007]Mokia 摩基亚

    Description: 摩尔瓦多的移动电话公司摩基亚(Mokia)设计出了一种新的用户定位系统.和其他的定位系统一样,它能够迅速回答任何形如"用户C的位置在哪?"的问题,精确到毫 ...

  8. cogs1752[boi2007]mokia 摩基亚 (cdq分治)

    [题目描述] 摩尔瓦多的移动电话公司摩基亚(Mokia)设计出了一种新的用户定位系统.和其他的定位系统一样,它能够迅速回答任何形如“用户C的位置在哪?”的问题,精确到毫米.但其真正高科技之处在于,它能 ...

  9. 【cdq分治】【P4390】[BOI2007]Mokia 摩基亚

    Description 给你一个 \(W~\times~W\) 的矩阵,每个点有权值,每次进行单点修改或者求某子矩阵内权值和,允许离线 Input 第一行是两个数字 \(0\) 和矩阵大小 \(W\) ...

随机推荐

  1. python全栈开发- 前⽅⾼能-迭代器

    python_day_12 今日主要内容 1, 函数名的应用,第一类对象 函数名可以像变量一样进行使用 1.赋值 2.作为list元素 3.作为参数 4.作为返回值 2, 闭包 内部函数访问外部函数的 ...

  2. 「Leetcode」976. Largest Perimeter Triangle(C++)

    分析 好久不刷题真的思维僵化,要考虑到这样一个结论:如果递增的三个数\(x_i,x_{i+1},x_{i+2}\)不符合题意,那么最大的两边之差一定大于等于第一条边,那么任何比第一条边小的都不能成立. ...

  3. JAVA Date、String、Calendar类型之间的转化

    1.Calendar 转化 String //获取当前时间的具体情况,如年,月,日,week,date,分,秒等 Calendar calendat = Calendar.getInstance(); ...

  4. 概念这种东西--node.js

    概念是一个既简单又复杂.既招人爱又招人恨的东西.概念是对一事务或现象的抽象.抽象好了,那就太方便问题的解决了,抽象坏了,那就驴唇不对马嘴,反而会让逻辑一塌糊涂.现实中经常有这样的概念:东北人怎么怎么样 ...

  5. centos下安装升级python到python3.5

    本文摘抄自:https://www.cnblogs.com/edward2013/p/5289056.html  请支持原版 CentOS7安装Python3.5   2. 安装Python的依赖包 ...

  6. Alpha发布——美工+文案

    此作业要求参见:https://edu.cnblogs.com/campus/nenu/2018fall/homework/2283 一.功能介绍 本团队(可以低头,但没必要)开发的是一款基于腾讯微信 ...

  7. 【分层最短路】Magical Girl Haze

    https://nanti.jisuanke.com/t/31001 有K次机会可以让一条边的权值变为0,求最短路. 在存储单源最短路的数组上多开一维状态,d[i][k]表示走到序号i的点,且让k条边 ...

  8. HDU 5172 GTY's gay friends 线段树+前缀和+全排列

    题目链接: hdu: http://acm.hdu.edu.cn/showproblem.php?pid=5172 bc(中文):http://bestcoder.hdu.edu.cn/contest ...

  9. <浪潮之巅>读书笔记

    <浪潮之巅>这本书通过介绍AT&T.IBM.微软.苹果.google等IT公司的发展历史,揭示科技工业的胜败规律,说明这些公司是如何在每一次科技革命浪潮到来时站在浪尖,实现跨越式发 ...

  10. SQL Server数据库复制

    事务复制 事务复制是一种复制类型,对订阅服务器上应用的初始数据快照,然后当发布服务器上发生数据修改时,将捕获到个别的事务并传播到订阅服务. 事务复制的原理是先将发布服务器数据库中的初始快照发送到各订阅 ...