POJ2528Mayor's posters(离散化 + 线段树)
题意:给定一些高度都相同的海报去贴,问最后能看见几张海报
The picture below illustrates the case of the sample input. 
{ 8,9,10}那张被覆盖看不到,
分析:看了大神们的博客大神是一看就是线段树,可是我都知道是线段树了也不知道怎么做,真是弱的一逼;
线段树分析就是对于每一个海报所在的区间设置一个不同的数字,然后就统计有几个不同的数字即所求解,就是线段树区间更改
技巧:离散化
如果按照这个墙的宽度来设置线段树的话,很大,内存会超。考虑到海报的个数不会很多,所以根据海报的区间的两个端点来进行离散处理,
拿例子来说,{1,4} {2,6} {8,10} {3,4} {7,10} ,所以需要的点为 1, 2, 3, 4, 6, 7, 8 ,10,令x[1] = 1, x[2] = 2,x[3] = 3,x[4] = 4,x[5] = 6,x[6] = 7,x[7] = 8, x[8] = 10,就将原来大的 【 1,10】区间变成了【1,8】这就是离散化
还有一个bug,如果这组样例: {1,10} {1,3} {5,10} 就离散成了 x[1] = 1, x[2] = 3, x[3] = 5, x[4] = 10,
第一张海报时候{1,10},对于离散化后的处理时,{1,4}
第二张{1,3}, 对于离散化后的就是{1,2}
第三张{5,10},对于离散化后的就是{3,4}
对于离散化后的来统计,最后结果是2,而实际确实3,原因就是 3到5之间有4而离散化之后就把4给去掉了,这就是一个bug,处理方法就是把相邻的两个数如果差大于1就在添加一个数
#include <iostream>
#include <algorithm>
#include <cstring>
#include <cstdio>
using namespace std;
const int Max = ; //根据海报的个数来设置数组
struct node
{
int l,r;
int cover; //海报的编号
int tag; //懒惰标志
};
node tree[Max * ];
int t[ * Max + ],cnt,res[Max * + ];
int a[Max + ],b[Max + ];
void buildTree(int left, int right, int k)
{
tree[k].l = left;
tree[k].r = right;
tree[k].cover = ;
tree[k].tag = -;
if(left == right)
return;
int mid = (left + right) / ;
buildTree(left, mid, k * );
buildTree(mid + , right, k * + );
}
int Find(int l, int r, int key, int temp[])
{
while(r >= l)
{
int mid = (l + r) / ;
if(temp[mid] == key)
return mid;
else if(temp[mid] > key)
{
r = mid - ;
}
else
{
l = mid + ;
}
}
return -;
}
void upDate(int left, int right, int k, int num)
{
if(tree[k].l == left && tree[k].r == right)
{
tree[k].cover = num;
tree[k].tag = num;
return;
}
if(tree[k].tag != -)
{
tree[k * ].tag = tree[k * + ].tag = tree[k].tag;
tree[k * ].cover = tree[k * + ].cover = tree[k].tag;
tree[k].tag = -;
}
int mid = (tree[k].l + tree[k].r) / ;
if(right <= mid)
{
upDate(left, right, k * , num);
}
else if(mid < left)
{
upDate(left, right, k * + , num);
}
else
{
upDate(left, mid, k * , num);
upDate(mid + , right, k * + , num);
}
}
void Search(int k)
{
if(tree[k].tag != -)
{
tree[k].cover = tree[k].tag;
tree[k * ].tag = tree[k * + ].tag = tree[k].tag;
tree[k * ].cover = tree[k * + ].cover = tree[k].tag;
tree[k].tag = -;
} if(tree[k].l == tree[k].r)
{
res[ tree[k].cover ] = ; // 看看每一个点的海报编号
return;
}
Search(k * );
Search(k * + );
}
int main()
{
int test,n;
scanf("%d", &test);
while(test--)
{
scanf("%d", &n);
cnt = ;
for(int i = ; i <= n; i++)
{
scanf("%d%d", &a[i], &b[i]);
t[ cnt++ ] = a[i];
t[ cnt++ ] = b[i]; //将所有的端点都放入t中
}
sort(t, t + cnt);
int cnt1 = ;
for(int i = ; i < cnt; i++)
{
if(t[i] != t[i - ]) //去掉相同的
t[ cnt1++ ] = t[i];
}
for(int i = cnt1 - ; i > ; i--)
{
if(t[i] - t[i - ] > ) //如果相邻数差值大于1,就则加一个数
t[ cnt1++ ] = t[i - ] + ;
}
sort(t, t + cnt1); //加完之后排序,此时已经离散完毕
for(int i = cnt1; i > ; i--)
t[i] = t[i - ];
//使区间为1到cnt1,好处理
buildTree(, cnt1, ); for(int i = ; i <= n; i++)
{
int left = Find(, cnt1, a[i], t); //二分查找出每个区间的端点在离散化后的数组中的位置
int right = Find(, cnt1, b[i], t);
upDate(left, right, , i); //更新,设置覆盖位为 i
}
memset(res, , sizeof(res));
Search(); //从根开始查找
int ans = ;
for(int i = ; i <= n; i++)
if(res[i])
ans++;
printf("%d\n", ans);
}
return ;
}
POJ2528Mayor's posters(离散化 + 线段树)的更多相关文章
- 【POJ】2528 Mayor's posters ——离散化+线段树
Mayor's posters Time Limit: 1000MS Memory Limit: 65536K Description The citizens of Bytetown, A ...
- Mayor's posters(离散化线段树)
Mayor's posters Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 54067 Accepted: 15713 ...
- POJ 2528 Mayor's posters 离散化+线段树
题目大意:给出一些海报和贴在墙上的区间.问这些海报依照顺序贴完之后,最后能后看到多少种海报. 思路:区间的范围太大,然而最多仅仅会有10000张海报,所以要离散化. 之后用线段树随便搞搞就能过. 关键 ...
- 南阳理工 题目9:posters(离散化+线段树)
posters 时间限制:1000 ms | 内存限制:65535 KB 难度:6 描述 The citizens of Bytetown, AB, could not stand that ...
- Mayor's posters (离散化线段树+对lazy的理解)
题目 题意: n(n<=10000) 个人依次贴海报,给出每张海报所贴的范围 li,ri(1<=li<=ri<=10000000) .求出最后还能看见多少张海报. 思路: 由于 ...
- SGU 180 Inversions(离散化 + 线段树求逆序对)
题目链接:http://acm.sgu.ru/problem.php?contest=0&problem=180 解题报告:一个裸的求逆序对的题,离散化+线段树,也可以用离散化+树状数组.因为 ...
- hpu校赛--雪人的高度(离散化线段树)
1721: 感恩节KK专场——雪人的高度 时间限制: 1 Sec 内存限制: 128 MB 提交: 81 解决: 35 [提交][状态][讨论版] 题目描述 大雪过后,KK决定在春秋大道的某些区间 ...
- 【BZOJ1645】[Usaco2007 Open]City Horizon 城市地平线 离散化+线段树
[BZOJ1645][Usaco2007 Open]City Horizon 城市地平线 Description Farmer John has taken his cows on a trip to ...
- 【bzoj4636】蒟蒻的数列 离散化+线段树
原文地址:http://www.cnblogs.com/GXZlegend/p/6801379.html 题目描述 蒟蒻DCrusher不仅喜欢玩扑克,还喜欢研究数列 题目描述 DCrusher有一个 ...
- 离散化+线段树/二分查找/尺取法 HDOJ 4325 Flowers
题目传送门 题意:给出一些花开花落的时间,问某个时间花开的有几朵 分析:这题有好几种做法,正解应该是离散化坐标后用线段树成端更新和单点询问.还有排序后二分查找询问点之前总花开数和总花凋谢数,作差是当前 ...
随机推荐
- 高仿Windows Phone QQ登录界面
给 TextBox文本框前添加图片 扩展PhoneTextBox:添加一个类"ExtentPhoneTextBox"继承 PhoneTextBox ,在"ExtentPh ...
- 3Dmax 创建物体
扩展基本体-切角长方体: 增加边: 删除边:在边选择模式下, 选择想要删除的边, 按下ctrl+backsapce
- [转]Win2003打不开https的问题
转自:http://ljl-xyf.iteye.com/blog/2269834 碰到客户做问题是能打开https://www.baidu.com 这个网页 打不开 https://sha256.al ...
- jQuery问题集锦
[1]阻止提交表单 方法1: $(function () { $("input[type=submit]").click(function (event) { //如果不满足表单提 ...
- coursera 公开课 文本挖掘和分析(text mining and analytics) week 1 笔记
一.课程简介: text mining and analytics 是一门在coursera上的公开课,由美国伊利诺伊大学香槟分校(UIUC)计算机系教授 chengxiang zhai 讲授,公开课 ...
- vim 插件管理
1 进入自己的vim mkdir ./bundle/vundle 2 在vimrc同级中执行 git clone https://github.com/gmarik/vundle.git ./bund ...
- useradd 添加用户
功能介绍 useradd命令用于Linux中创建的新的系统用户.useradd可用来建立用户帐号.帐号建好之后,再用passwd设定帐号的密码.而可用userdel删除帐号.使用useradd指令所建 ...
- 【web端】百度地图api
百度地图js加载 页面引入: 修改申请的密钥:<script type="text/javascript" src="http://api.map.baidu.co ...
- Java Web项目中的经典代码抽取
前言: 众所周知的,项目开发中做得最多的无非就是增删查改(CRUD)操作.自从国内Web项目开发渐渐盛行SSH框架之后,其开发开发流程也变得更加灵活:本文就项目开发中的业务层代码作个简单的抽取,供业内 ...
- url 字符串中的参数信息
/// <summary> /// 分析 url 字符串中的参数信息 /// </summary> /// <param nam ...