POJ 2528 Mayor‘s poster 线段树+离散化
给一块最大为10^8单位宽的墙面,贴poster,每个poster都会给出数据 a,b,表示该poster将从第a单位占据到b单位,新贴的poster会覆盖旧的,最多有10^4张poster,求最后贴完,会看到几张poster (哪怕只露出一个单位,也算该poster可见);
我一看这么大数据,又看了下时间限制只有1s,不科学啊,如果真的按10^8建树不可能过时间啊,而且根据它的空间限制,大概只能建10^7这么大的数组。
后来搜博客发现大家的标题都写着离散化,原来用离散化做这个题目,但是我不会离散化,我想找一篇纯讲离散化的博文来好好研究下,。。。没找到,所以原谅我,这个题目是仔仔细细的分析了别人的原题代码才写出来的,否则我真的不知道怎么弄离散化。
所谓离散化,把a,b所有出现的数字全部排序一遍,新的值就是他们的下标,当然重复的不算,然后每个a,b就会赋予一个更小的值,但完全不影响其功能,由于每个不同值都考虑进去了,所以区间缩小之后,不会出现覆盖混乱,最多是正好全部覆盖。。。。
然后说起这个线段树要维护的,其实就是一个量,该区间是否被覆盖,采取从后往前贴poster,这个跟之前做个一个排队插队的题目同理,从后往前,则状态一定固定了,所以更加方便了,只要判断在已经覆盖了这么多poster的情况下,是否还有空余区间就行了。
话说期间WA了几次不明不白是因为,我忘了用懒惰标记和懒惰更新了,果然好久没写线段树了。。。
#include <iostream>
#include <cstdio>
#include <cstring>
#include <algorithm>
#define N 25010
#define Lson x<<1,l,mid
#define Rson (x<<1|1),mid+1,r using namespace std; struct node {
int val;
int num;
} pos[N];
bool flag;
bool d[N*];
int p[N*];
bool cmp2(node a,node b)
{
if (a.val<b.val) return true;
return false;
}
bool cmp(node a,node b)
{
if (a.num>b.num) return true;
if (a.num==b.num && a.val<b.val) return true;
return false;
}
void getup(int x)
{
if (d[x<<]==true || d[x<<|]==true)
d[x]=true;
else
d[x]=false;
}
void build(int x,int l,int r)
{
d[x]=true;
p[x]=;
if (l==r){
return;
}
int mid=(l+r)>>;
build(Lson);
build(Rson);
getup(x);
}
void pushdown(int x,int l,int r)
{
if (!p[x] || l>=r) return;
d[x<<]=d[x<<|]=false;
p[x<<]=p[x<<|]=p[x];
p[x]=;
}
void query(int L,int R,int x,int l,int r)
{
if (L<=l && r<=R)
{
if (d[x])
{
flag=true;
d[x]=false;
p[x]=;
}
return;
}
if (p[x])
pushdown(x,l,r);
int mid=(l+r)>>;
if (L>mid && d[x<<|]) query(L,R,Rson);
else
if (R<=mid && d[x<<]) query(L,R,Lson);
else
if (L<=mid && R>mid){
if (d[x<<]) query(L,R,Lson);
if (d[x<<|]) query(L,R,Rson);
}
getup(x);
}
int main()
{
int t;
scanf("%d",&t);
while (t--)
{
int n;
int i,j,k;
scanf("%d",&n);
int a,b;
for (i=;i<n;i++)
{
scanf("%d%d",&pos[i<<].val,&pos[i<<|].val);
pos[i<<].num=pos[i<<|].num=i;
}
sort(pos,pos+*n,cmp2);
int cur=,pre=-;
for (j=;j<*n;j++)
{
if (pre!=pos[j].val){
pre=pos[j].val;
pos[j].val=++cur;
}
else{
pos[j].val=cur;
}
//cout<<pre<<" "<<pos[j].val<<endl;
} build(,,cur); sort(pos,pos+*n,cmp);
int ans=;
for (i=;i<n;i++)
{
//cout<<pos[i<<1].val<<" "<<pos[i<<1|1].val<<endl;
flag=false;
query(pos[i<<].val,pos[i<<|].val,,,cur);
if (flag) {ans++;}
}
printf("%d\n",ans);
}
return ;
}
POJ 2528 Mayor‘s poster 线段树+离散化的更多相关文章
- poj 2528 Mayor's posters 线段树+离散化技巧
poj 2528 Mayor's posters 题目链接: http://poj.org/problem?id=2528 思路: 线段树+离散化技巧(这里的离散化需要注意一下啊,题目数据弱看不出来) ...
- POJ 2528 Mayor's posters(线段树+离散化)
Mayor's posters 转载自:http://blog.csdn.net/winddreams/article/details/38443761 [题目链接]Mayor's posters [ ...
- poj 2528 Mayor's posters 线段树+离散化 || hihocode #1079 离散化
Mayor's posters Description The citizens of Bytetown, AB, could not stand that the candidates in the ...
- POJ 2528 Mayor's posters (线段树+离散化)
Mayor's posters Time Limit: 1000MS Memory Limit: 65536K Total Submissions:75394 Accepted: 21747 ...
- POJ.2528 Mayor's posters (线段树 区间更新 区间查询 离散化)
POJ.2528 Mayor's posters (线段树 区间更新 区间查询 离散化) 题意分析 贴海报,新的海报能覆盖在旧的海报上面,最后贴完了,求问能看见几张海报. 最多有10000张海报,海报 ...
- POJ 2528 Mayor's posters (线段树区间更新+离散化)
题目链接:http://poj.org/problem?id=2528 给你n块木板,每块木板有起始和终点,按顺序放置,问最终能看到几块木板. 很明显的线段树区间更新问题,每次放置木板就更新区间里的值 ...
- POJ 2528 Mayor’s posters (线段树段替换 && 离散化)
题意 : 在墙上贴海报, n(n<=10000)个人依次贴海报,给出每张海报所贴的范围li,ri(1<=li<=ri<=10000000).求出最后还能看见多少张海报. 分析 ...
- poj 2528 Mayor's posters 线段树区间更新
Mayor's posters Time Limit: 1 Sec Memory Limit: 256 MB 题目连接 http://poj.org/problem?id=2528 Descript ...
- poj 2528 Mayor's posters(线段树)
题目:http://poj.org/problem?id=2528 题意:有一面墙,被等分为1QW份,一份的宽度为一个单位宽度.现在往墙上贴N张海报,每张海报的宽度是任意的, 但是必定是单位宽度的整数 ...
随机推荐
- Java日志相关概述
日志是代码调试.生产运维必备工具,基本所有软件都会有日志记录. 1.常用日志框架介绍 1.Logging jdk1.5自带日志工具类,位于java.util.logging; 2.Log4j 市场占有 ...
- 019、MySQL取本季度开始时间和本季度结束时间
SELECT QUARTER ( adddate( dy, ) ) QTR, date_add( dy, INTERVAL MONTH ) Q_start, adddate( dy, ) Q_end ...
- LeetCode160 相交链表(双指针)
题目: click here!!题目传送门 思路: 1.笨方法 因为如果两个链表相交的话,从相交的地方往后是同一条链表,所以: 分别遍历两个链表,得出两个链表的长度,两个长度做差得到n,然后将长的链表 ...
- POJ3295 Tautology重言式
Tautology 思路很简单,对于p.q.r.s.t暴力枚举是0还是1,判断即可.判断时像写表达式求值那样用栈.为了方便可以从后往前,因为最后一个肯定不是运算.那几个奇奇怪怪的函数可以找规律然后转为 ...
- 洛谷 P2549 计算器写作文
题目传送门 解题思路: 背包,f[i]表示计算器位数为i时,可获得的最大分值. 本题与01背包不同的地方在于,物品的摆放顺序对答案是有影响的,例如两个字符串a,b,那么就会出现a+b和b+a两种情况( ...
- python面试题整理(二)
1.进程,线程,协程定义,有什么区别 进程是操作系统分配资源的最小单位,一个进程对应一块CPU 线程是进程中的某一个控制单元,是CPU调度的最小单元,线程之间相互独立,进程结束线程也会结束,一个进程至 ...
- 051-PHP求余运算
<?php $x=10%5; //进行求余运算 $y=10%3; //进行求余运算 $z=10%6; //进行求余运算 echo $x; //输出变量x的值 echo $y; //输出变量y的值 ...
- 了解C#
了解C C#能编写那些程序 Windows桌面应用程序 桌面应用有自己独立的进程与操作系统进行消息通讯,操作系统对事件进行检测,传递给桌面应用进程,桌面应用进程对这些消息进行解释,处理后,把处理结果u ...
- VUE- Cordova打包APP
VUE- Cordova打包APP 现在使用vue开发的项目越来越多,使用vue开发的移动端打包就成了最大的问题.现在前端打包方案有好多种,但是综合来说,我比较喜欢用cordova来进行Android ...
- 16 协程和www
1.一个应用程序一般对应一个进程,一个进程一般有一个主线程,还有若干个辅助线程,线程之间是平行运行的,在线程里面可以开启协程,让程序在特定的时间内运行.2协程和线程的区别是:协程避免了无意义的调度,由 ...