线段树入门(Billboard)

Billboard

Time Limit:8000MS     Memory
Limit:
32768KB    
64bit IO Format:%I64d & %I64u

Description

At the entrance to the university, there is a huge
rectangular billboard of size h*w (h is its height and w is its width). The
board is the place where all possible announcements are posted: nearest
programming competitions, changes in the dining room menu, and other important
information.

On September 1, the billboard was empty. One by one, the
announcements started being put on the billboard.

Each
announcement is a stripe of paper of unit height. More specifically, the i-th
announcement is a rectangle of size 1 * wi.

When someone puts a
new announcement on the billboard, she would always choose the topmost possible
position for the announcement. Among all possible topmost positions she would
always choose the leftmost one.

If there is no valid location for
a new announcement, it is not put on the billboard (that‘s why some programming
contests have no participants from this university).

Given the
sizes of the billboard and the announcements, your task is to find the numbers
of rows in which the announcements are placed.

Input

There are multiple cases (no more than 40 cases).


The first line of the input file contains three integer numbers, h,
w, and n (1 <= h,w <= 10^9; 1 <= n <= 200,000) - the dimensions of
the billboard and the number of announcements.

Each of the next n
lines contains an integer number wi (1 <= wi <= 10^9) - the width of i-th
announcement.

Output

For each announcement (in the order they are given
in the input file) output one number - the number of the row in which this
announcement is placed. Rows are numbered from 1 to h, starting with the top
row. If an announcement can‘t be put on the billboard, output "-1" for this
announcement.

Sample Input

3 5 5

2

4

3

3

Sample
Output

1

2

1

3

-1

线段树的水题。。

【题目大意】

有一个高和宽分别为h和w的广告牌,初始时牌子为空,然后在上面贴广告,每条广告的高都为1,宽为Wi,只能横着贴,位置的优先级为:上->左,输入一系列广告的宽,输出该条广告锁在的行数,如果贴不下了就输出-1。

【题目分析】

首先分析题目的思路,一开始可能会想到很多的方法,其实暴力模拟也可以,但是看看题目的数据:200,000,暴力模拟的话时间复杂度O(n^2),妥妥的TLE了,所以要用到树状数组来优化时间复杂度,线段树的功能是维护区间的最大值,即:query:区间求最大值的位子(直接把update的操作在query里做了),也就是每一行剩余的空格的最大值。

分析到这儿,题目就变得简单多了。

source code:


#include<iostream>
#include<cstdio>
#include<cstring>
#include<cstdlib>
#define MAX 200010
using namespace std;
int h,w,n,a;
struct Node
{
int l,r;
int len;
} Tree[MAX*4];

void build(int l,int r,int x)
{
Tree[x].l=l;
Tree[x].r=r;
Tree[x].len=w;//初始化的时候空格的长度都是w
if(Tree[x].l==Tree[x].r)
return;
int mid=(l+r)>>1;
build(l,mid,2*x);
build(mid+1,r,2*x+1);
Tree[x].len=Tree[2*x].len>Tree[2*x+1].len?Tree[2*x].len:Tree[2*x+1].len;//更新最大值
}
int query(int num,int x)
{
if(Tree[x].l==Tree[x].r)
{
Tree[x].len-=num;//更新线段树最底层的值
return Tree[x].l;//返回行数
}
int ans;
if(num<=Tree[2*x].len)//小于左子树的长度,访问左子树
ans=query(num,2*x);
else if(num<=Tree[2*x+1].len)
ans=query(num,2*x+1);
Tree[x].len=Tree[2*x].len>Tree[2*x+1].len?Tree[2*x].len:Tree[2*x+1].len;//向上更新父节点的值
return ans;
}
int main()
{
while(scanf("%d%d%d",&h,&w,&n)!=EOF)
{
a=h<n?h:n;//只需要处理n和h中最小的行数
build(1,a,1);
int x;
while(n--)
{
scanf("%d",&x);
if(x>Tree[1].len) //如果输入的广告长度比剩余最大的宽度还要大,则无解
printf("-1\n");
else
printf("%d\n",query(x,1));
}
}
return 0;
}

线段树入门(Billboard),码迷,mamicode.com

时间: 2024-10-11 06:10:51

线段树入门(Billboard)的相关文章

线段树入门小结

QUE:线段树? 称谓: 从刘汝佳的书中得知,"这种数据结构在学术界没有统一的术语,但线段树是最常见的叫法.其他叫法包括区间树(interval tree).范围树(range tree)等,但这些属于在特定的场合(如计算几何)中有着特殊的意义".怎么叫看读者的心情,以下统一用线段树称呼. 先来作一些了解: 线段树是一棵二叉树,它的左右儿子也都是一棵线段树.(定义) 线段树也叫区间树,为什么叫它区间树呢?因为线段树是一种基于区间的数据结构. 线段树的每个节点代表一个区间 [L,R],其

线段树入门(I Hate It)

I Hate It Time Limit:3000MS     Memory Limit:32768KB     64bit IO Format:%I64d & %I64u Description 很多学校流行一种比较的习惯.老师们很喜欢询问,从某某到某某当中,分数最高的是多少. 这让很多学生很反感. 不管你喜不喜欢,现在需要你做的是,就是按照老师的要求,写一个程序,模拟老师的询问.当然,老师有时候需要更新某位同学的成绩. Input 本题目包含多组测试,请处理到文件结束. 在每个测试的第一行,

《数据结构》线段树入门(二)

今天继续介绍——线段树之延迟标记 接上期<数据结构>线段树入门(一):http://www.cnblogs.com/shadowland/p/5870339.html 在上期介绍了线段树的最基本内容(线段树单点修改,区间查询),这次将介绍:区间修改,区间查询. Question: 给你N个数,有两种操作: 1:给区间[a,b]的所有数增加X 2:询问区间[a,b]的数的和. 输入描述: 第一行一个正整数n,接下来n行n个整数,再接下来一个正整数Q,每行表示操作的个数,如果第一数是1,后接3个正

线段树入门理解

在复习算法至分治法时,书本上主要介绍了合并排序和快速排序,较为简单.特拓展简单学习一个应用了分治法的算法结构--线段树. acm刷题时遇到许多连续区间的动态查询问题,例如求取某一区间上元素之和.求取某一区间上元素的最大值,此时如果使用一般的方法求解会使得时间超出要求.此时需要使用到线段树,其主要用于高效解决连续区间的动态查询问题. 线段树,类似区间树,是一个完全二叉树,它在各个节点保存一条线段(数组中的一段子数组),由于二叉结构的特性,它基本能保持每个操作的复杂度为O(lgN),从而大大减少耗时

hdu1166敌兵布阵&amp;&amp;hdu1754I Hate It(线段树入门)

单点更新是最最基础的线段树,只更新叶子节点,然后把信息用pushup这个函数更新上来. http://acm.hdu.edu.cn/showproblem.php?pid=1166 update单点更新,query区域求和. #include <iostream> #include <stdio.h> #include <string.h> #include <stdlib.h> #define N 200001 using namespace std; s

hdu 1754 I Hate It(线段树入门)

I Hate It Time Limit: 9000/3000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total Submission(s): 41510    Accepted Submission(s): 16458 Problem Description 很多学校流行一种比较的习惯.老师们很喜欢询问,从某某到某某当中,分数最高的是多少. 这让很多学生很反感. 不管你喜不喜欢,现在需要你做的是,就是按照老师

线段树入门(更新单个节点)

很多学校流行一种比较的习惯.老师们很喜欢询问,从某某到某某当中,分数最高的是多少. 这让很多学生很反感. 不管你喜不喜欢,现在需要你做的是,就是按照老师的要求,写一个程序,模拟老师的询问.当然,老师有时候需要更新某位同学的成绩. Input本题目包含多组测试,请处理到文件结束. 在每个测试的第一行,有两个正整数 N 和 M ( 0<N<=200000,0<M<5000 ),分别代表学生的数目和操作的数目. 学生ID编号分别从1编到N. 第二行包含N个整数,代表这N个学生的初始成绩,

线段树入门---给定多个线段求点的出现个数

线段树是一颗二叉树,他的每个节点都是一个区间,此题为线段树的入门题目,只是学习笔记.例题:给定N个线段,给定M个点,求点在多少个线段中出现过,此时如果用传统的方法来求,时间复杂度太高,但是,线段树的时间复杂度还可以接受. 步骤为: 1. 首先找一个区间,能覆盖给定的所有区间, 然后把此区间建立线段树,建立线段树的方式是二分法建立,即它的左孩子是他的左半个区间,右孩子是它的右边那个区间.一个图足以说明清楚 2. 将所有的区间映射到此树上, 从根节点开始遍历, 每遍历一个节点考虑四种情况: 1) 当

线段树入门题 hdu1166 敌兵布阵

题意:动态查询一段区间的和,支持单点更新. 开始刷线段树了,这是入门第一题...最基本的线段树操作,递归建树,递归查询,递归修改,向上更新节点,每个节点的值代表该节点对应区间的和 . 代码: