题意:有一块宣传栏,高一定,给出长度,再给出多张海报的张贴位置,问还能见到几张海报(哪怕有一点被看到)?假设海报的高于宣传栏同高。

思路:问题转成“给出x轴上长为L的一条线段,再用n条线段进行覆盖上去,最后还能看到及条线”。长度是0~L,即长度是L,进行离散化的时候,应该用1~L,每个数字表示一个单位长。还有就是按照提示所给的信息实现即可。步骤如下:

(1)保存n个数据,做成pair,并将所有出现过的数字在另外找地方排序,去掉重复的,再将数据紧缩化处理,那么大小在1~max。再将紧缩化的数据与原数据做成映射表。

(2)建立线段树,原始n个数据进行映射后对线段树插入更新。注意上限的问题,因为已经离散化了。每个节点上有个值val,表示处于此段区间的是编号为val的海报。

(3)DFS搜索线段树,遇到tag=1的节点就进行标记,然后返回,其子树都没用处的,已经被覆盖。可能会有重复的,比如一段为[2,3],一段为[3,5],因为不能放在一起,所以分开放的时候会有重复的val,要去重。

 #include <bits/stdc++.h>
using namespace std;
const int N=;
int n,l,rr,ll,cnt=;
bool vis[N];
unordered_map<int,int> mapp;
pair<int,int> pai[N];
struct node
{
int val;
bool tag;
node *ll,*rr;
}; node *create()
{
node *tmp=new(node);
tmp->val=tmp->tag=;
tmp->ll=tmp->rr=;
return tmp;
} void update(int l,int r,int LL,int RR,int num,node *t) //其实可以不用递归,因为不用回溯
{
if(l==LL&&r==RR)//找到了对应区间
{
t->tag=;
t->val=num;
return ;
}
if( LL!=RR && !t->ll ) //有孩子,但是还没有建立
{
t->ll=create();
t->rr=create();
t->ll->tag= t->rr->tag= ;
t->ll->val= t->rr->val= t->val;
} int mid=((LL+RR)>>);
if(t->tag)//需要对此节点下推
{
if(t->ll)//前提是有孩子
{
t->ll->tag= t->rr->tag= ;
t->ll->val= t->rr->val= t->val;
}
t->tag=;
} if(l>mid) update(l,r,mid+,RR,num,t->rr);//在右边
else if(r<=mid) update(l,r,LL,mid,num,t->ll);//在左边
else
{
update(l,mid,LL,mid,num,t->ll);
update(mid+,r,mid+,RR,num,t->rr);
}
} void DFS(int LL,int RR,node *t) //进行深搜,搜到tag为1的就有用,可能存在重复
{
if(t->tag)
{
vis[t->val]=;//为了去重
return;
}
int mid=((LL+RR)>>);
DFS(LL,mid,t->ll);
DFS(mid+,RR,t->rr);
} int main()
{
//freopen("input.txt", "r", stdin); cin>>n>>l;
vector<int> vect; //辅助
for(int i=; i<n; i++)
{
scanf("%d%d",&ll,&rr);
pai[i]=make_pair(ll,rr);
vect.push_back(ll);
vect.push_back(rr);
} sort(vect.begin(),vect.end());
vect.push_back(-); //仅仅为了下面的for不会溢出。
int up=;
for(int i=;i<vect.size()-; i++) //紧缩化处理
{
if(vect[i]!=vect[i+])
mapp[vect[i]]=++up; //做个哈希表
}
vect.clear(); node *tree=create(); //建树根节点
tree->tag=; for(int i=; i<n; i++) //根据pair进行修改树。设置tag,在需要修改其儿子时在进行下推。
{
int L=mapp[pai[i].first];
int R=mapp[pai[i].second];
update(L,R-,,up-,i+,tree); //插入修改
} DFS(,up,tree);//深搜
for(int i=; i<=n; i++) if(vis[i]) cnt++;
cout<<cnt<<endl; return ;
}

AC代码

hihoCoder #1079 : 离散化 (线段树,数据离散化)的更多相关文章

  1. ACM/ICPC 之 数据结构-线段树+区间离散化(POJ2528)

    这道题用线段树做更方便更新和查询,但是其数据范围很大,因此要将离散化和线段树结合起来,算是一道比较经典的线段树+离散化的例题. 线段树的离散化有很多方法,在这里,我先用一次结点离散化,间接将源左右端点 ...

  2. 【POJ】2528 Mayor's posters ——离散化+线段树

    Mayor's posters Time Limit: 1000MS    Memory Limit: 65536K   Description The citizens of Bytetown, A ...

  3. hpu校赛--雪人的高度(离散化线段树)

    1721: 感恩节KK专场——雪人的高度 时间限制: 1 Sec  内存限制: 128 MB 提交: 81  解决: 35 [提交][状态][讨论版] 题目描述 大雪过后,KK决定在春秋大道的某些区间 ...

  4. 离散化+线段树/二分查找/尺取法 HDOJ 4325 Flowers

    题目传送门 题意:给出一些花开花落的时间,问某个时间花开的有几朵 分析:这题有好几种做法,正解应该是离散化坐标后用线段树成端更新和单点询问.还有排序后二分查找询问点之前总花开数和总花凋谢数,作差是当前 ...

  5. poj/OpenJ_Bailian - 2528 离散化+线段树

    传送门:http://bailian.openjudge.cn/practice/2528?lang=en_US //http://poj.org/problem?id=2528 题意: 给你n长海报 ...

  6. 南阳理工 题目9:posters(离散化+线段树)

    posters 时间限制:1000 ms  |  内存限制:65535 KB 难度:6   描述 The citizens of Bytetown, AB, could not stand that ...

  7. SGU 180 Inversions(离散化 + 线段树求逆序对)

    题目链接:http://acm.sgu.ru/problem.php?contest=0&problem=180 解题报告:一个裸的求逆序对的题,离散化+线段树,也可以用离散化+树状数组.因为 ...

  8. hdu1542 Atlantis (线段树+扫描线+离散化)

    Atlantis Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) Total S ...

  9. 【BZOJ1645】[Usaco2007 Open]City Horizon 城市地平线 离散化+线段树

    [BZOJ1645][Usaco2007 Open]City Horizon 城市地平线 Description Farmer John has taken his cows on a trip to ...

  10. 【bzoj4636】蒟蒻的数列 离散化+线段树

    原文地址:http://www.cnblogs.com/GXZlegend/p/6801379.html 题目描述 蒟蒻DCrusher不仅喜欢玩扑克,还喜欢研究数列 题目描述 DCrusher有一个 ...

随机推荐

  1. 深度学习之softmax回归

    前言            以下内容是个人学习之后的感悟,转载请注明出处~ softmax回归 首先,我们看一下sigmod激活函数,如下图,它经常用于逻辑回归,将一个real value映射到(0, ...

  2. 使用IntelliJ IDEA配置Tomcat(入门)

    一.下载Tomcat 1.进入官网http://tomcat.apache.org/,选择download,下载所需Tomcat版本. 此处我们选择下载最新版本Tomcat 9. 注意有zip和exe ...

  3. Eigen::Map

    http://cherishlc.iteye.com/blog/2116800 Map类 是 矩阵库Eigen中用来将内存数据 映射为 任意形状的矩阵的类.

  4. 【Data Structure & Algorithm】求1+2+…+n

    求1+2+-+n 题目:求1+2+-+n,要求不能使用乘除法.for.while.if.else.switch.case等关键字以及条件判断语句(A ? B : C). 分析:此题没多少实际意义,因为 ...

  5. 技术胖Flutter第四季-20导航的参数传递和接受-1

    技术胖Flutter第四季-20导航的参数传递和接受-1 视频地址:https://www.bilibili.com/video/av35800108/?p=21 先安装一个新的插件: Awesome ...

  6. CSS3:nth-child()伪类选择器,Table表格奇偶数行定义样式

    转自爱设计 原文链接http://www.dangshopex.com/jishufenxiang/WEBkaifajishu/8653.html CSS3的强大,让人惊叹,人们在惊喜之余,又不得不为 ...

  7. java日期时间处理集合

    本文主要介绍java中日期时间的处理,包括获取时间,时间相加减,格式化等操作. 持续更新中... 时间格式化 //时间格式化 SimpleDateFormat dateFormat = new Sim ...

  8. hdu2147(yy)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=2147 题意:给出一个n*m的矩阵,一开始有个点在最右上方, 两个人轮流移动点,可以向左移一格,或者向下 ...

  9. 洛谷P2485 [SDOI2011]计算器(exgcd+BSGS)

    传送门 一题更比三题强 1操作直接裸的快速幂 2操作用exgcd求出最小正整数解 3操作用BSGS硬上 然后没有然后了 //minamoto #include<cstdio> #inclu ...

  10. 笔记-JavaWeb学习之旅10

    Servlet server applet运行在服务器端的小程序,servlet就是一个接口,定义了Java类被浏览器访问到的规则(Java类重写这个接口,就可以被浏览器(tomcat)识别) Ser ...