POJ 3416 Crossing --离线+树状数组
题意: 给一些平面上的点,然后给一些查询(x,y),即以(x,y)为原点建立坐标系,一个人拿走第I,III象限的点,另一个人拿II,IV象限的,点不会在任何一个查询的坐标轴上,问每次两人的点数差为多少。
解法:离线树状数组。点不在坐标轴上,即点不共线使这题简单了不少,可以离散化点,也可以不离散化,因为x,y <= 500000,直接就可以搞。我这里是离散的,其实也没比直接搞快。
见两个树状数组,一个先把所有点都modify进去,一个等待以后加元素。
然后将查询和给出的点都按y坐标排序,然后离线对每个查询执行操作了。每次查询前把y坐标小于当前查询点的点加入树状数组。
这时的 左下角点数即为: LD = getsum(c2,Q[i].x-1);
右上角: UR = getsum(c1,maxi)-getsum(c1,Q[i].x)-(getsum(c2,maxi)-getsum(c2,Q[i].x)); 即为整个右边的个数减去y坐标小于此点的(即为右下角)。
那么另两个象限的综述就是 n-LD-UR。
这样就解决了。
代码:
#include <iostream>
#include <cstdio>
#include <cstring>
#include <cstdlib>
#include <cmath>
#include <algorithm>
using namespace std;
#define N 100007 struct node{
int x,y,ind;
}p[N],Q[N];
int n,m,maxi;
int c1[N],c2[N],a[N],b[N],ans[N];
int ma[],mb[];
int lowbit(int x) { return x&-x; }
int cmp(node ka,node kb) { return ka.y < kb.y; } void modify(int *c,int x,int val)
{
while(x <= maxi)
c[x] += val, x += lowbit(x);
} int getsum(int *c,int x)
{
int res = ;
while(x > ) { res += c[x]; x -= lowbit(x); }
return res;
} int main()
{
int i,j,t;
scanf("%d",&t);
while(t--)
{
scanf("%d%d",&n,&m);
for(i=;i<=n;i++)
{
scanf("%d%d",&p[i].x,&p[i].y);
a[i] = p[i].x, b[i] = p[i].y;
}
for(i=;i<=m;i++)
{
scanf("%d%d",&Q[i].x,&Q[i].y);
a[i+n] = Q[i].x, b[i+n] = Q[i].y;
Q[i].ind = i;
}
sort(a+,a+n+m+);
sort(b+,b+n+m+);
int inda = unique(a+,a+n+m+)-a-;
int indb = unique(b+,b+n+m+)-b-;
maxi = max(inda,indb);
for(i=;i<=inda;i++) ma[a[i]] = i;
for(i=;i<=indb;i++) mb[b[i]] = i; for(i=;i<=n;i++) p[i].x = ma[p[i].x], p[i].y = mb[p[i].y];
for(i=;i<=m;i++) Q[i].x = ma[Q[i].x], Q[i].y = mb[Q[i].y];
sort(p+,p+n+,cmp);
sort(Q+,Q+m+,cmp);
memset(c1,,sizeof(c1));
memset(c2,,sizeof(c2));
for(i=;i<=n;i++)
modify(c1,p[i].x,);
j = ;
for(i=;i<=m;i++)
{
while(j <= n && p[j].y <= Q[i].y)
modify(c2,p[j].x,), j++;
int LD = getsum(c2,Q[i].x-);
int UR = getsum(c1,maxi)-getsum(c1,Q[i].x)-(getsum(c2,maxi)-getsum(c2,Q[i].x));
ans[Q[i].ind] = abs(*(LD+UR)-n);
}
for(i=;i<=m;i++) printf("%d\n",ans[i]);
if(t >= ) puts("");
}
return ;
}
直接搞不离散的代码:
#include <iostream>
#include <cstdio>
#include <cstring>
#include <cstdlib>
#include <cmath>
#include <algorithm>
using namespace std;
#define N 500007 struct node{
int x,y,ind;
}p[N],Q[N];
int n,m,maxi;
int c1[N],c2[N],a[N],b[N],ans[N];
int lowbit(int x) { return x&-x; }
int cmp(node ka,node kb) { return ka.y < kb.y; } void modify(int *c,int x,int val)
{
while(x <= maxi)
c[x] += val, x += lowbit(x);
} int getsum(int *c,int x)
{
int res = ;
while(x > ) { res += c[x]; x -= lowbit(x); }
return res;
} int main()
{
int i,j,t;
scanf("%d",&t);
while(t--)
{
scanf("%d%d",&n,&m);
for(i=;i<=n;i++)
{
scanf("%d%d",&p[i].x,&p[i].y);
p[i].x++, p[i].y++;
maxi = max(maxi,p[i].x);
}
for(i=;i<=m;i++)
{
scanf("%d%d",&Q[i].x,&Q[i].y);
Q[i].x++, Q[i].y++;
Q[i].ind = i;
maxi = max(maxi,Q[i].x);
}
sort(p+,p+n+,cmp);
sort(Q+,Q+m+,cmp);
memset(c1,,sizeof(c1));
memset(c2,,sizeof(c2));
for(i=;i<=n;i++)
modify(c1,p[i].x,);
j = ;
for(i=;i<=m;i++)
{
while(j <= n && p[j].y <= Q[i].y)
modify(c2,p[j].x,), j++;
int LD = getsum(c2,Q[i].x-);
int UR = getsum(c1,maxi)-getsum(c1,Q[i].x)-(getsum(c2,maxi)-getsum(c2,Q[i].x));
ans[Q[i].ind] = abs(*(LD+UR)-n);
}
for(i=;i<=m;i++) printf("%d\n",ans[i]);
if(t >= ) puts("");
}
return ;
}
POJ 3416 Crossing --离线+树状数组的更多相关文章
- HDU 2852 KiKi's K-Number(离线+树状数组)
题目链接 省赛训练赛上一题,貌似不难啊.当初,没做出.离线+树状数组+二分. #include <cstdio> #include <cstring> #include < ...
- CF #365 (Div. 2) D - Mishka and Interesting sum 离线树状数组
题目链接:CF #365 (Div. 2) D - Mishka and Interesting sum 题意:给出n个数和m个询问,(1 ≤ n, m ≤ 1 000 000) ,问在每个区间里所有 ...
- CF #365 (Div. 2) D - Mishka and Interesting sum 离线树状数组(转)
转载自:http://www.cnblogs.com/icode-girl/p/5744409.html 题目链接:CF #365 (Div. 2) D - Mishka and Interestin ...
- HDU3333 Turing Tree 离线树状数组
题意:统计一段区间内不同的数的和 分析:排序查询区间,离线树状数组 #include <cstdio> #include <cmath> #include <cstrin ...
- 离线树状数组 hihocoder 1391 Countries
官方题解: // 离线树状数组 hihocoder 1391 Countries #include <iostream> #include <cstdio> #include ...
- 13年山东省赛 Boring Counting(离线树状数组or主席树+二分or划分树+二分)
转载请注明出处: http://www.cnblogs.com/fraud/ ——by fraud 2224: Boring Counting Time Limit: 3 Sec ...
- 区间的关系的计数 HDU 4638 离线+树状数组
题目大意:给你n个人,每个人都有一个id,有m个询问,每次询问一个区间[l,r],问该区间内部有多少的id是连续的(单独的也算是一个) 思路:做了那么多离线+树状数组的题目,感觉这种东西就是一个模板了 ...
- BZOJ_2743_[HEOI2012]采花_离线+树状数组
BZOJ_2743_[HEOI2012]采花_离线+树状数组 Description 萧芸斓是Z国的公主,平时的一大爱好是采花.今天天气晴朗,阳光明媚,公主清晨便去了皇宫中新建的花园采花 .花园足够大 ...
- SPOJ DQUERY - D-query (莫队算法|主席树|离线树状数组)
DQUERY - D-query Given a sequence of n numbers a1, a2, ..., an and a number of d-queries. A d-query ...
随机推荐
- 欧拉计划之题目9:找出唯一的满足a + b + c = 1000的毕达哥拉斯三元组{a, b, c}
本题来自:http://pe.spiritzhang.com/index.php/2011-05-11-09-44-54/10-9a--b--c--1000a-b-c #include <std ...
- [Xamarin.Android] Support Library Tips
[Xamarin.Android] Support Library Tips Support Library支持内容 Xamarin Support Library每个版本支持.那些组件,可以参考这份 ...
- new 小记
new运算符 能根据需求来创建对象的实例 通过与构造函数和一系列初始化过程中使用的可选参数来创建对象的实例,对象创建完成后,新创建的对象继承自构造函数的原型 function Person(name) ...
- 趣味题:恺撒Caesar密码(c++实现)
描述:Julius Caesar 生活在充满危险和阴谋的年代.为了生存,他首次发明了密码,用于军队的消息传递.假设你是Caesar 军团中的一名军官,需要把Caesar 发送的消息破译出来.并提供给你 ...
- HTML中行内元素的竖直方向的padding和margin是否真的无效
参考资料:Inline elements and padding 今天写一个导航栏时遇到了一个问题:行内元素的padding-top,padding-bottom和margin-top,margin- ...
- vIDC v2.0 强大的端口转发神器使用总结-开放内网tfs代码服务
vIDC2.0 端口映射工具,最近在公司闲来无事,想自己整个tfs来管理自己的研究代码. 本来是想用微软Visual Studio提供的免费tfs,但是无奈速度太慢.他们的服务器在美国,中国也没有代理 ...
- SarePoint Powershell Add user to Group
$FromGroupnames = "001总经理","010101管理本部" $ToGroupname = "test" $SPWeb = ...
- HBase读写路径的工作机制
出处:http://wuyudong.com/1946.html HBase 写路径工作机制 在HBase 中无论是增加新行还是修改已有的行,其内部流程都是相同的.HBase 接到命令后存下变化信息, ...
- OC知识梳理-NSArray与NSMutableArray相关知识
知识普及: 1.数组中的元素在系统中都会有其默认对应的下标,下标是一个整形的数字,默认从0开始. 例:NSArray *arr3 = @["345","234" ...
- iOS设计模式之代理模式
代理模式 基本理解 代理模式(Proxy),为其他对象提供一种代理以控制对这个对象的访问. 代理模式的应用 远程代理:就是为一个对象在不同的地址空间提供据不代表.这样可以隐藏一个对象存在于不同地址空间 ...