经典线段树的题。

每个节点存储的信息:左端点连续空房间的长度,右端点连续空房间长度,连续空房间的最大长度。

由于要求每次必须从尽量靠左边的位置进行居住,那么搜索时应尽量让区间起始位置更小:

1.如果当前区间的最大空房间长度小于要寻找的长度,说明不会找到符合的区间,直接退出。

2.如果左子区间的最大空房间长度大于等于要寻找的长度,那么应该进入左子区间查找。

3.否则,如果左子区间的右端点连续空房间长度 + 右子区间的左端点连续空房间的长度 大于等于
要寻找的长度则直接返回左子区间右端点起始空房间下标。

4.否则,进入右区间寻找。

AC代码:

#include<cstdio>
#include<algorithm>
using namespace std;
const int maxn = 2e5;
struct node{
	int l, r;
	int len, lenl, lenr;
	int tag; //0代表全空  1代表全满 -1代表混杂
}t[maxn];

void build(int l, int r, int cur){
	t[cur].l = l, t[cur].r = r;
	t[cur].len = t[cur].lenl = t[cur].lenr = r - l + 1;
	t[cur].tag = 0;
	if(l == r) return;
	int mid = (l + r) / 2;
	build(l, mid, cur << 1);
	build(mid + 1, r, (cur << 1) + 1);
}

void lazy(int cur){
	if(t[cur].l == t[cur].r) return;
	if(t[cur].tag >= 0) {  //覆盖
		int l = cur << 1, r = (cur << 1) + 1;
		int val;
		if(t[cur].tag == 0) val = t[l].r - t[l].l + 1;
		else val = 0;
		t[l].len = t[l].lenl = t[l].lenr = val;

		if(t[cur].tag == 0) val = t[r].r - t[r].l + 1;
		else val = 0;
		t[r].len = t[r].lenl = t[r].lenr = val;
		t[r].tag = t[l].tag = t[cur].tag;
	}
}

int search(int len, int cur){
	if(t[cur].len < len) return 0;
	int l = t[cur].l, r = t[cur].r;
	if(l == r) return l;
	lazy(cur);
	if(t[cur << 1].len >= len) return search(len, cur << 1);
	else if(t[cur << 1].lenr + t[(cur << 1) + 1].lenl >= len) return t[cur << 1].r - t[cur << 1].lenr + 1;
	else if(t[(cur << 1) + 1].len >= len) return search(len, (cur << 1) +1);
}

void update(int l, int r, int tag, int cur){
	int l1 = t[cur].l, r1 = t[cur].r;
	if(l1 == l && r1 == r){
		t[cur].tag = tag;
		int val;
		if(tag) val = 0;
		else val = r1 - l1 + 1;
		t[cur].len = t[cur].lenl = t[cur].lenr = val;
		return;
	}
	lazy(cur);
	int mid = (l1 + r1) / 2;
	if(r <= mid) update(l, r, tag, cur << 1);
	else if(l >= mid + 1) update(l, r, tag, (cur << 1) + 1);
	else {
		update(l, mid, tag, cur << 1);
		update(mid + 1, r, tag, (cur << 1) + 1);
	}

	int c1 = cur << 1, c2 = (cur << 1) + 1;
	int w = t[c1].lenr + t[c2].lenl;
	t[cur].lenl = t[c1].lenl, t[cur].lenr = t[c2].lenr;
	if(t[c1].tag == 0) t[cur].lenl = w;
	if(t[c2].tag == 0) t[cur].lenr = w;
	t[cur].len = max(t[c1].len, t[c2].len);
	t[cur].len = max(w, t[cur].len);
	t[cur].len = max(t[cur].len, t[cur].lenr);
	t[cur].len = max(t[cur].len, t[cur].lenl);
	if(t[cur].len == 0) t[cur].tag = 1;
	else if(t[cur].len == r1 - l1 + 1) t[cur].tag = 0;
	else t[cur].tag = -1;
}

int main(){
	int n,q;
	while(scanf("%d%d", &n, &q) == 2){
		build(1, n, 1);
		int a, b, c;
		for(int i = 0; i < q; ++i){
			scanf("%d", &a);
			if(a == 1){
				scanf("%d", &b);
				int x = search(b, 1);
				printf("%d\n", x);
				if(x != 0) update(x, x + b - 1, 1, 1);
			}
			else {
				scanf("%d%d", &b, &c);
				update(b, b + c - 1, 0, 1);
			}
		}
	}
	return 0;
}

如有不当之处欢迎指出!

poj Hotel 线段树的更多相关文章

  1. poj 2886 线段树+反素数

    Who Gets the Most Candies? Time Limit: 5000MS   Memory Limit: 131072K Total Submissions: 12744   Acc ...

  2. [USACO08FEB]酒店Hotel 线段树

    [USACO08FEB]酒店Hotel 线段树 题面 其实就是区间多维护一个lmax,rmax(表示从左开始有连续lmax个空房,一直有连续rmax个空房到最右边),合并时讨论一下即可. void p ...

  3. poj 3667 Hotel (线段树)

    http://poj.org/problem?id=3667 Hotel Time Limit: 3000MS   Memory Limit: 65536K Total Submissions: 94 ...

  4. POJ 3667 Hotel(线段树 区间合并)

    Hotel 转载自:http://www.cnblogs.com/scau20110726/archive/2013/05/07/3065418.html [题目链接]Hotel [题目类型]线段树 ...

  5. POJ 1823 Hotel 线段树

    题目链接 线段树的区间合并. 和上一题差不多....第三种操作只需要输出maxx[1]的值就可以. #include <iostream> #include <vector> ...

  6. poj 3468(线段树)

    http://poj.org/problem?id=3468 题意:给n个数字,从A1 …………An m次命令,Q是查询,查询a到b的区间和,c是更新,从a到b每个值都增加x.思路:这是一个很明显的线 ...

  7. POJ——3264线段树

    题目: 输入两个数(m,n),m表示牛的头数,n表示查询的个数.查询时输入两个数(x,y),表示查询范围的起始值和终止值,查询结果是,这个区间内牛重量的最大值减去牛重量的最小值,数量级为1000,00 ...

  8. POJ 2828 线段树(想法)

    Buy Tickets Time Limit: 4000MS   Memory Limit: 65536K Total Submissions: 15422   Accepted: 7684 Desc ...

  9. PKU 3667 Hotel(线段树)

    Hotel The cows are journeying north to Thunder Bay in Canada to gain cultural enrichment and enjoy a ...

随机推荐

  1. MySql按日期进行统计

    1 数据库字段pk_time(Varchar) 当天的数据 SELECT * FROM 表 WHERE date(fk_time) = curdate(); 当月的数据 SELECT *FROM 表 ...

  2. 02_HTML5+CSS详解第四天

    依旧是CSS部分贴个CSS主要知识点总结的链接:http://blog.csdn.net/html5_/article/details/26098273 [自己的笔记做得好乱,以前一直以为是字丑的原因 ...

  3. js调用系统虚拟键盘

    <input type="text" id="tt" /> <script language="javascript" t ...

  4. Win10图片打开方式没有“Windows照片查看器”,如何找回?

    如果你是全新安装的Win10正式版,那么就会发现当在图片上点击右键时,"打开方式"菜单里熟悉的"Windows照片查看器"不见了,换成了Win10全新的&quo ...

  5. 关于if和else嵌套—蛋疼

    嵌套使用的时候else if和else遵循就近原则,和上面最靠近该语句的if语句匹配,要把else if看成是一个整体.就这么干,这样好理解一点. 一个if...else if...else语句中可以 ...

  6. Storm 对 0.10.x 版 Kafka之commit offsets

    由于 0.10.x 版 Kafka 与 0.8.x 版有很大的变化,这种变化对下游 Storm 有非常大的影响,0.10.x 版的 Kafka 不但增加了权限管理的功能,而且还将 simple 和 h ...

  7. 给HttpClient添加Socks代理

    本文描述http client使用socks代理过程中需要注意的几个方面:1,socks5支持用户密码授权:2,支持https:3,支持让代理服务器解析DNS: 使用代理创建Socket 从原理上来看 ...

  8. python socket单线程通信

    注意 socket通信默认的情况下是属于阻塞形式通信,在recv与accept函数会进行阻塞 1.客户端正常退出的情况下,server端的con的通道会正常的关闭,recv直接赋值为空 2.在wind ...

  9. 数据分区------《Designing Data-Intensive Applications》读书笔记9

    进入到第六章了,我们要开始聊聊分布式系统之中的核心问题:数据分区.分布式系统通常是通过大规模的数据节点来处理单机没有办法处理的海量数据集,因此,可以将一个大型数据集可以分布在多个磁盘上,查询负载可以分 ...

  10. MOBA 游戏技能系统设计 2.0

    随着游戏开发的完整度提升,技能系统的设计复杂性也越来越高,导致了用模板方式的配置方法和处理方法会导致以下几个问题: 代码冗余 排错困难 配置项冗余 熟悉业务流程时间长 扩展性低 经过我思考决定重写之. ...