2016-08-15

题意:一面墙,往上面贴海报,后面贴的可以覆盖前面贴的。问最后能看见几种海报。

思路:可以理解成往墙上涂颜色,最后能看见几种颜色(下面就是以涂色来讲的)。这面墙长度为1~1000 0000,一千万,确实很大。暴力的话肯定不行,除非..( you know)。

正确的解法是用线段树,不过还得加上离散化,因为数据太大10000000啊。

先说一下离散化,这个其实就是压缩,把范围压缩,举个例子:

输入 :

1 3000    //涂第一种颜色 范围从1~10000      下面同理

2000 7000  //第二种颜色

我们可以先把输入的数据范围压缩,假如输入的数据存到数组a[0]=1,a[1]=3000,a[2]=2000,a[3]=7000。

离散化(压缩)过程:

先把a[]从小到大排序。然后接着

把a数组里的数据映射到dis数组里,这样涂颜色(插入)的时候用dis[a[i]]。

压缩完之后建树的时候只需建1~4范围的树,而不用建1~7000的树。所以说离散化是必须的...懂嘞谬?

离散化完事后,再说说线段树,到底是个什么树呢?

比如1~10这个线段,用二叉树的形式储存起来,

看图:

然后在每个节点加上各种信息,就可以用了,虽然空间大了但时间快了

对于这个问题 线段树主要用了三个函数

void build(int p,int a,int b) //建树

void update(int p,int col,int a,int b)//插入颜色

void query(int p)  //查询颜色

代码有详细解释

AC代码:

 #include <iostream>
#include <cstring>
#include <cstdio>
#include <algorithm>
using namespace std; const int MAXN=; bool tagcol[MAXN]; //标记颜色是否已经计算过 int e[MAXN<<]; //临时数组,把重复的数据去掉
int ee[MAXN][]; //直接输入数据的数组 存每张海报的范围的
int dis[]; //离散化之后的映射数组
int cnt=; //颜色计数器 struct node //线段树节点
{
int l,r; //l~r 范围
int c; //颜色(用1,2,3...表示)
}LTnode[MAXN<<]; //开数组的时候尽量开大点,开8倍就Runtime Error了,开16倍就过,虽然不知道为啥 void build(int p,int a,int b) //建树,p是节点下标,范围 a~b
{
LTnode[p].l=a;
LTnode[p].r=b;
LTnode[p].c=;
if(a==b) //到叶子节点直接返回
{
return;
} //继续递归建树
int mid=(a+b)>>;
build(p<<,a,mid);
build(p<<|,mid+,b);
return;
}
void update(int p,int col,int a,int b) //更新(插入)颜色,p是节点下标,col是颜色,a,b是要更新的范围
{
if(LTnode[p].l>=a&&LTnode[p].r<=b) //如果当前节点范围 正好在a,b内,更新颜色 返回
{
LTnode[p].c=col;
return;
}
if(LTnode[p].r<a||LTnode[p].l>b) //如果当前节点范围和a,b没有交集
return;
if(LTnode[p].c>=) //当前节点和a,b有部分交集
{
LTnode[p<<].c=LTnode[p<<|].c=LTnode[p].c; //把当前节点的颜色传递给左右孩子节点,灰常重要
LTnode[p].c=-; //-1代表有多种颜色
}
//继续处理左右孩子
update(p<<,col,a,b);
update(p<<|,col,a,b);
return;
} void query(int p) //查询,p是节点下标
{
if(LTnode[p].c==) //当前节点的颜色为0,即没有颜色
return;
if(LTnode[p].c>&&!tagcol[LTnode[p].c]) //当前节点有某种颜色并且该颜色没有被计算过
{
cnt++; //颜色计数器+1
tagcol[LTnode[p].c]=true;
return;
}
if(LTnode[p].c==-) //多色 则继续细化处理左右孩子
{
query(p<<);
query(p<<|);
}
return; } int main()
{
int T;
scanf("%d",&T);
while(T--)
{ memset(tagcol,false,sizeof(tagcol));
memset(dis,,sizeof(dis)); int n; //n表示有几张海报
//cin>>n;
scanf("%d",&n);
int maxp=;
for(int i=;i<n;i++)
{
//cin>>ee[i][0]>>ee[i][1];
scanf("%d%d",&ee[i][],&ee[i][]); //poj上用scanf就过,cin就TLE 坑啊...
if(!dis[ee[i][]]) //用dis数组先标记下 有没有重复的数据 这样就不用再开一个数组了
{
e[maxp++]=ee[i][];
dis[ee[i][]]=;
}
if(!dis[ee[i][]])
{
e[maxp++]=ee[i][];
dis[ee[i][]]=;
}
}
memset(dis,,sizeof(dis));
int has=;
sort(e,e+maxp); //先排序
for(int i=;i<maxp;i++) //for循环出来后 has 就是离散后的最大范围
{
dis[e[i]]=++has;
}
build(,,has);
for(int i=;i<n;i++) //一次插入颜色,i为颜色
{
update(,i+,dis[ee[i][]],dis[ee[i][]]);
}
cnt=;
query();
cout<<cnt<<endl; } return ;
}

手打真累啊,肩膀疼..

poj2528(线段树+离散化)Mayor's posters的更多相关文章

  1. POJ 2528 (线段树 离散化) Mayor's posters

    离散化其实就是把所有端点放在一起,然后排序去个重就好了. 比如说去重以后的端点个数为m,那这m个点就构成m-1个小区间.然后给这m-1个小区间编号1~m-1,再用线段树来做就行了. 具体思路是,从最后 ...

  2. poj2528 线段树+离散化 (倒序)

    The citizens of Bytetown, AB, could not stand that the candidates in the mayoral election campaign h ...

  3. 【线段树】Mayor's posters

    [poj2528]Mayor's posters Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 66154   Accept ...

  4. POJ2528 线段树离散化

    Mayor's posters Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 62771   Accepted: 18120 ...

  5. 线段树 G - Mayor's posters 小技巧

    G - Mayor's posters POJ - 2528 这个题目要倒着来写,从后面往前面贴,因为前面的有些会被后面的覆盖. 所以我们就判断这张海报的位置有没有完全被覆盖,如果完全被覆盖了就不能贴 ...

  6. poj2528 线段树+离散化

    由于坐标可能很大,此时需要离散化,将值转化为对应的坐标. #include<stdio.h> #include<algorithm> using namespace std; ...

  7. (线段树)Mayor's posters --poj -- 2528

    链接: http://poj.org/problem?id=2528 覆盖问题, 要从后往前找, 如果已经被覆盖就不能再覆盖了,否则就可以覆盖 递归呀递归什么时候我才能吃透你 代码: #include ...

  8. [poj2528] Mayor's posters (线段树+离散化)

    线段树 + 离散化 Description The citizens of Bytetown, AB, could not stand that the candidates in the mayor ...

  9. POJ 2528 Mayor's posters(线段树+离散化)

    Mayor's posters 转载自:http://blog.csdn.net/winddreams/article/details/38443761 [题目链接]Mayor's posters [ ...

随机推荐

  1. Codeforces Beta Round #97 (Div. 1)

    B 判矩阵的时候 出了点错 根据点积判垂直 叉积判平行 面积不能为0 #include <iostream> #include<cstdio> #include<cstr ...

  2. 使用Java API创建(create),查看(describe),列举(list),删除(delete)Kafka主题(Topic)

    使用Kafka的同学都知道,我们每次创建Kafka主题(Topic)的时候可以指定分区数和副本数等信息,如果将这些属性配置到server.properties文件中,以后调用Java API生成的主题 ...

  3. (转载)C语言 数组与指针的区别

    1) 字符串指针变量是个变量,指向字符串的首地址:而字符串数组名是个常量,为字符串数组第一个元素的地址: 2)字符串指针变量可以赋值,而字符串数组名不能赋值:对于字符数组只能对各个元素赋值,不能用以下 ...

  4. Cookie存储中文报错:java.lang.IllegalArgumentException: Control character in cookie value or attribute.(转)

    项目中做自动登录和保存密码时,Cookie报错Java.lang.IllegalArgumentException,上google查了下 在http://hi.baidu.com/xtxycy/blo ...

  5. S2sh整合MAven项目所需坐标大全

    <dependency> <groupId>junit</groupId> <artifactId>junit</artifactId> & ...

  6. UVa 11464 Even Parity 偶数矩阵

    给你一个 n * n 的 01 矩阵,现在你的任务是将这个矩阵中尽量少的 0 转化为 1 ,使得每个数的上下左右四个相邻的数加起来是偶数.求最少的转化个数. 首先,n 的规模并不大,最大只有15.但是 ...

  7. Android源码分析--CircleImageView 源码详解

    源码地址为 https://github.com/hdodenhof/CircleImageView 实际上就是一个圆形的imageview 的自定义控件.代码写的很优雅,实现效果也很好, 特此分析. ...

  8. Hadoop1.1.2伪分布式安装笔记

    一.设置Linux的静态IP 修改桌面图标修改,或者修改配置文件修改 1.先执行ifconfig,得到网络设备的名称eth0 2.编辑/etc/sysconfig/network-scripts/if ...

  9. 【转】IOS --- OC与Swift混编

    群里大神发的网址,感觉有用就先收录了,暂时没时间看SWIFT,感觉代码简洁,但是可阅读性不是太高,有些代码让系统去判断类型,同样的,我们看代码的时候也得自己去判断类型,或许看多就习惯了,有时间再说吧, ...

  10. ylb:创建数据库、表,对表的增查改删语句

    ylbtech-SQL Server:SQL Server-创建数据库.表,对表的增查改删语句 SQL Server 创建数据库.表,对表的增查改删语句. 1,ylb:创建数据库.表,对表的增查改删语 ...