HDU 2795 Billboard (线段树+贪心)
手动博客搬家:本文发表于20170822 21:30:17, 原地址https://blog.csdn.net/suncongbo/article/details/77488127
URL: http://acm.hdu.edu.cn/showproblem.php?pid=2795题目大意:有一个hw的木板 (h, w<=1e9), 现在有n (n<=2e5)张1xi的海报要贴在木板上,按1~n的顺序每次贴海报时会选择最上的一排的最左边贴 (海报不能互相覆盖), 求每张海报会被贴在哪一行。最上方的一行编号为1, 以此类推。
样例解析:
如下
column 12345
row 1: 11333
row 2: 2222X
row 3: 444XXX: 未摆放;
对应行列上的数是摆放在此的海报的编号思路分析:本题实际上是一个贪心的思想。
每次可以从最上一行向下枚举剩余的空间,找到第一个能够贴上第i个海报的行,然后输出,更新即可。
时间复杂度O(n^2), 无法AC.
于是我们可以考虑二分的思想:
用线段树维护每一行剩余的空间的大小的最大值。
每次查询时,采用类似于二分答案的方法,每到达线段树的一个节点,若它的左子树最大值>=xi, 则左子树中必存在合法的最大答案,查询左子树;
假如左子树中的行无法装下第i张海报,但右子树最大值<=xi, 此时说明右子树中存在合法的最大答案,于是“退而求其次”,查询右子树。
如果两棵子树最大值都>xi, 则无法张贴此海报,于是输出-1.
查询完毕后,做一个单点修改,将答案所在的行剩余空间减去xi.
其实,最后一种情况不需要考虑。
假如整个[1, h]区间的最大值<xi, 则直接输出-1, 无需查询。
但是我们无法开4e9 (4h)如此大的数组,怎么办呢?
其实假如h>n, 那么下面的(h-n)行全部浪费了,不影响结果。
因此,h=min(h,n);
只开8e5 (4n)的数组即可。
代码呈现:(Time: 3151 MS; 11256 KB; Code: 1576 B)
#include<cstdio>
#include<algorithm>
using namespace std;
const int MAXN = 2e5;
struct Node
{
int left,right;
int maxi;
};
struct SegmentTree
{
Node nd[MAXN*4+8];
void init()
{
for(int i=1; i<=MAXN*4; i++)
{
nd[i].left = nd[i].right = nd[i].maxi = 0;
}
}
void build(int lbound,int rbound,int pos,int tot)
{
nd[pos].left = lbound;
nd[pos].right = rbound;
if(lbound==rbound)
{
nd[pos].maxi = tot;
return;
}
int mid = (lbound+rbound)/2;
build(lbound,mid,2*pos,tot);
build(mid+1,rbound,2*pos+1,tot);
nd[pos].maxi = tot;
}
void modify_subs(int bound,int val,int pos)
{
int mid = (nd[pos].left+nd[pos].right)/2;
if(nd[pos].left==nd[pos].right)
{
nd[pos].maxi-=val;
return;
}
if(bound<=mid) modify_subs(bound,val,2*pos);
else modify_subs(bound,val,2*pos+1);
nd[pos].maxi = max(nd[2*pos].maxi,nd[2*pos+1].maxi);
}
int query(int pos,int val)
{
int mid = (nd[pos].left+nd[pos].right)/2;
if(nd[pos].left==nd[pos].right) return nd[pos].left;
int ans;
if(val<=nd[2*pos].maxi) return query(2*pos,val);
else return query(2*pos+1,val);
}
};
SegmentTree sgt;
int n,m,p;
int main()
{
while(scanf("%d%d%d",&n,&m,&p)!=EOF)
{
if(n>p) n = p;
sgt.init();
sgt.build(1,n,1,m);
for(int i=1; i<=p; i++)
{
int x;
scanf("%d",&x);
if(sgt.nd[1].maxi < x)
{
printf("-1\n");
continue;
}
int ans = sgt.query(1,x);
printf("%d\n",ans);
if(ans>0) sgt.modify_subs(ans,x,1);
}
}
return 0;
}
HDU 2795 Billboard (线段树+贪心)的更多相关文章
- hdu 2795 Billboard 线段树单点更新
Billboard Time Limit: 1 Sec Memory Limit: 256 MB 题目连接 http://acm.hdu.edu.cn/showproblem.php?pid=279 ...
- HDU 2795 Billboard (线段树)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=2795 题目大意:有一块h*w的矩形广告板,要往上面贴广告; 然后给n个1*wi的广告,要求把广告贴 ...
- [HDU] 2795 Billboard [线段树区间求最值]
Billboard Time Limit: 20000/8000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Total ...
- HDU 2795 Billboard 线段树,区间最大值,单点更新
Billboard Time Limit: 20000/8000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) Total ...
- ACM学习历程—HDU 2795 Billboard(线段树)
Description At the entrance to the university, there is a huge rectangular billboard of size h*w (h ...
- HDU 2795 Billboard (线段树单点更新 && 求区间最值位置)
题意 : 有一块 h * w 的公告板,现在往上面贴 n 张长恒为 1 宽为 wi 的公告,每次贴的地方都是尽量靠左靠上,问你每一张公告将被贴在1~h的哪一行?按照输入顺序给出. 分析 : 这道题说明 ...
- HDU 2795 Billboard 线段树活用
题目大意:在h*w 高乘宽这样大小的 board上要贴广告,每个广告的高均为1,wi值就是数据另给,每组数组给了一个board和多个广告,要你求出,每个广告应该贴在board的哪一行,如果实在贴不上, ...
- hdu 2795 Billboard 线段树+二分
Billboard Time Limit: 20000/8000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) Probl ...
- HUD.2795 Billboard ( 线段树 区间最值 单点更新 单点查询 建树技巧)
HUD.2795 Billboard ( 线段树 区间最值 单点更新 单点查询 建树技巧) 题意分析 题目大意:一个h*w的公告牌,要在其上贴公告. 输入的是1*wi的w值,这些是公告的尺寸. 贴公告 ...
随机推荐
- 【Ubuntu】基本操作 (条目=11)
定义 NAME 为要操作的对象名 定义 DIR 为文件所在的绝对路径 所有操作默认在普通用户下进行 所有软件包默认是指Debian包(deb包) 1.查看进程 top 2.强制结束进程 PID由top ...
- C# Socket 您的主机中的软件中止了一个已建立的连接 An established connection was aborted by the software in your host machine
http://tieba.baidu.com/p/3223234493 问题: 服务端在接收客户端数据的时候,抛了个异常出来:System.Net.Sockets.SocketException: 您 ...
- Spring Boot、微服务架构和大数据
一文读懂 Spring Boot.微服务架构和大数据治理三者之间的故事 https://www.cnblogs.com/ityouknow/p/9034377.html 微服务架构 微服务的诞生并非偶 ...
- thinkphp data方法
data方法也是模型类的连贯操作方法之一,用于设置当前要操作的数据对象的值,可能大家不太习惯用这个方法,今天来讲解下如何用好data方法. 用法 写操作 通常情况下我们都是通过create方法或者赋值 ...
- ASP.NET Core 多环境
ASP.NET Core 支持在多个环境中管理应用程序,如开发(Development),预演(Staging)和生产(Production).环境变量用来指示应用程序正在运行的环境,允许应用程序适当 ...
- 知识总结1:debug命令汇总(一)
-r 得到当前寄存器状态或修改寄存器状态 -d 跳到默认(CS:IP指定的地址)区间 -d 段地址:偏移地址 跳到(段地址*+偏移地址)区间 -d 段地址:起始偏移地址 终止偏移地址 跳到(段地址*+ ...
- POJ 1330 Tarjan LCA、ST表(其实可以数组模拟)
题意:给你一棵树,求两个点的最近公共祖先. 思路:因为只有一组询问,直接数组模拟好了. (写得比较乱) 原题请戳这里 #include <cstdio> #include <bits ...
- css中background-clip属性的作用
background-clip属性的通俗作用就是指定元素背景所在的区域,有四种取值 1.border-box border-box是默认值,表示元素的背景从border区域(包括border)以内开始 ...
- Ubuntu16下安装lamp
1.安装php7 sudo apt-get install php7.0 php7.0-mcrypt 2.安装MySQL sudo apt-get install mysql-server 输入 su ...
- 时序分析:ARIMA模型(非平稳时间序列)
转载于一篇硕士论文.... ARIMA模型意为求和自回归滑动平均模型(IntergratedAut少regressive MovingAverageModel),简记为ARIMA(p,d,q),p,q ...