[USACO08FEB]Hotel 题解
正确的题解
首先我们都知道这题要用线段树做。考虑维护靠左边的answer,靠右边的answer,和整个区间的answer,那么就珂以维护这道题目了。
这里比较复杂的有下传操作和上传操作。
上传
void pushUp(int pos, int l, int r){
seg[pos].ans = max(seg[pos << 1].rans + seg[pos << 1 | 1].lans, max(seg[pos << 1].ans, seg[pos << 1 | 1].ans));
int mid = (l + r) >> 1;
if (seg[pos << 1].ans == mid - l + 1)
seg[pos].lans = seg[pos << 1 | 1].lans + seg[pos << 1].ans;
else
seg[pos].lans = seg[pos << 1].lans;
if (seg[pos << 1 | 1].ans == r - mid)
seg[pos].rans = seg[pos << 1].rans + seg[pos << 1 | 1].ans;
else
seg[pos].rans = seg[pos << 1 | 1].rans;
}
下传
void pushDown(int pos, int l, int r){
if (!seg[pos].lazy){
seg[pos << 1].lazy = seg[pos << 1 | 1].lazy = 0;
int mid = (l + r) >> 1;
seg[pos << 1].ans = seg[pos << 1].lans = seg[pos << 1].rans = mid - l + 1;
seg[pos << 1 | 1].ans = seg[pos << 1 | 1].lans = seg[pos << 1 | 1].rans = r - mid;
}
else if (seg[pos].lazy == 1){
seg[pos << 1].lazy = seg[pos << 1 | 1].lazy = 1;
seg[pos << 1].ans = seg[pos << 1].lans = seg[pos << 1].rans = 0;
seg[pos << 1 | 1].ans = seg[pos << 1 | 1].lans = seg[pos << 1 | 1].rans = 0;
}
seg[pos].lazy = -1;
}
正解代码
#include <cstdio>
#include <algorithm>
using namespace std;
int read(){
int x = 0; int zf = 1; char ch = ' ';
while (ch != '-' && (ch < '0' || ch > '9')) ch = getchar();
if (ch == '-') zf = -1, ch = getchar();
while (ch >= '0' && ch <= '9') x = x * 10 + ch - '0', ch = getchar(); return x * zf;
}
struct Node{
int ans, lans, rans;
int lazy;
} seg[200000];
void pushUp(int pos, int l, int r){
seg[pos].ans = max(seg[pos << 1].rans + seg[pos << 1 | 1].lans, max(seg[pos << 1].ans, seg[pos << 1 | 1].ans));
int mid = (l + r) >> 1;
if (seg[pos << 1].ans == mid - l + 1)
seg[pos].lans = seg[pos << 1 | 1].lans + seg[pos << 1].ans;
else
seg[pos].lans = seg[pos << 1].lans;
if (seg[pos << 1 | 1].ans == r - mid)
seg[pos].rans = seg[pos << 1].rans + seg[pos << 1 | 1].ans;
else
seg[pos].rans = seg[pos << 1 | 1].rans;
}
void pushDown(int pos, int l, int r){
if (!seg[pos].lazy){
seg[pos << 1].lazy = seg[pos << 1 | 1].lazy = 0;
int mid = (l + r) >> 1;
seg[pos << 1].ans = seg[pos << 1].lans = seg[pos << 1].rans = mid - l + 1;
seg[pos << 1 | 1].ans = seg[pos << 1 | 1].lans = seg[pos << 1 | 1].rans = r - mid;
}
else if (seg[pos].lazy == 1){
seg[pos << 1].lazy = seg[pos << 1 | 1].lazy = 1;
seg[pos << 1].ans = seg[pos << 1].lans = seg[pos << 1].rans = 0;
seg[pos << 1 | 1].ans = seg[pos << 1 | 1].lans = seg[pos << 1 | 1].rans = 0;
}
seg[pos].lazy = -1;
}
void build(int pos, int l, int r){
if (l == r){
seg[pos].ans = seg[pos].lans = seg[pos].rans = 1;
seg[pos].lazy = -1;
return ;
}
int mid = (l + r) >> 1;
build(pos << 1, l, mid), build(pos << 1 | 1, mid + 1, r);
pushUp(pos, l, r);
}
void modify(int pos, int l, int r, int x, int y, int val){
if (x <= l && r <= y){
if (val) seg[pos].ans = seg[pos].lans = seg[pos].rans = 0;
else seg[pos].ans = seg[pos].lans = seg[pos].rans = r - l + 1;
seg[pos].lazy = val;
return ;
}
pushDown(pos, l, r); int mid = (l + r) >> 1;
if (x <= mid) modify(pos << 1, l, mid, x, y, val);
if (y > mid) modify(pos << 1 | 1, mid + 1, r, x, y, val);
pushUp(pos, l, r);
}
int query(int pos, int l, int r, int k){
pushDown(pos, l, r); if (l == r) return l;
int mid = (l + r) >> 1;
if (seg[pos << 1].ans >= k) return query(pos << 1, l, mid, k);
if (seg[pos << 1].rans + seg[pos << 1 | 1].lans >= k) return (mid - seg[pos << 1].rans + 1);
else return query(pos << 1 | 1, mid + 1, r, k);
}
int main(){
int n = read(), m = read();
build(1, 1, n);
for (int i = 1; i <= m; ++i){
int op = read(), x, y;
if (op == 1){
x = read();
if(seg[1].ans >= x){
int l = query(1, 1, n, x);
printf("%d\n", l);
modify(1, 1, n, l, l + x - 1, 1);
}
else
puts("0");
}
else{
x = read(), y = read();
modify(1, 1, n, x, x + y - 1, 0);
}
}
return 0;
}
提供一种新的得高分思路
这是我发这篇题解的目的,我们看到这熟悉的区间推平操作,很容易就想到了珂朵莉树,也就是我一开始的打法,但是由于数据的原因珂树T了,伤心.jpg。
我们考虑珂朵莉树的查询,显然我们只需要更改一下查询即可
int query(int k){
int l = 1, cnt = 0;
for (set<Node>::iterator it = st.begin(); it != st.end(); ++it){
if (it->val == 1){
l = it->r + 1;
cnt = 0;
}
else{
cnt += it->r - it->l + 1;
if (cnt >= k) return l;
}
}
return -1;
}
我们考虑把连在一起的块的值为零的区间的长度加起来,取第一个满足条件的端点,然后就珂以做出来啦。
可怜的92分代码
#include <cstdio>
#include <vector>
#include <algorithm>
#include <set>
#include <map>
using namespace std;
int read(){
int x = 0; int zf = 1; char ch = ' ';
while (ch != '-' && (ch < '0' || ch > '9')) ch = getchar();
if (ch == '-') zf = -1, ch = getchar();
while (ch >= '0' && ch <= '9') x = x * 10 + ch - '0', ch = getchar(); return x * zf;
}
//build
struct Node{
int l, r;
mutable bool val;
Node(int a = -1, int b = -1, int c = 0){
l = a, r = b, val = c;
}
bool operator < (const Node &a) const{
return l < a.l;
}
};
set<Node> st;
//modify
set<Node>::iterator split(int pos){
set<Node>::iterator it = st.lower_bound(Node(pos));
if (it != st.end() && it->l == pos) return it;
--it; Node tmp = *it; st.erase(it);
st.insert(Node(tmp.l, pos - 1, tmp.val));
return st.insert(Node(pos, tmp.r, tmp.val)).first; //first return iterator
}
void assign(int l, int r, bool val){
set<Node>::iterator itr = split(r + 1), itl = split(l);
st.erase(itl, itr);
st.insert((Node){l, r, val});
}
//query
int query(int k){
int l = 1, cnt = 0;
for (set<Node>::iterator it = st.begin(); it != st.end(); ++it){
if (it->val == 1){
l = it->r + 1;
cnt = 0;
}
else{
cnt += it->r - it->l + 1;
if (cnt >= k) return l;
}
}
return -1;
}
int main(){
int n = read(), m = read(); st.insert((Node){1, n, 0});
while (m--){
int op = read();
if (op == 1){
int x = read(), pos = query(x);
if (pos == -1) puts("0");
else{
printf("%d\n", pos);
assign(pos, pos + x - 1, 1);
}
}
else if (op == 2){
int x = read(), y = read();
assign(x, x + y - 1, 0);
}
}
return 0;
}
[USACO08FEB]Hotel 题解的更多相关文章
- USACO08FEB Hotel
题目传送门 线段树维护区间 线段树结构体 struct zzz{ int l,r,mi; //l为以左端点的为起点的最长子串 //r为以右端点为终点的最长子串 //mi是区间内部的最长子串 }tree ...
- POJ3667 Hotel 题解
和最大子段和的思路是一样的,可以记 \(lmax,rmax,dat\) 分别表示从当前区间最靠左/右的最大连续空子段和当前区间的最大连续空子段. 需要用延迟标记,每次遇到开房操作先ask,如果能找到就 ...
- 线段树||BZOJ1593: [Usaco2008 Feb]Hotel 旅馆||Luogu P2894 [USACO08FEB]酒店Hotel
题面:P2894 [USACO08FEB]酒店Hotel 题解:和基础的线段树操作差别不是很大,就是在传统的线段树基础上多维护一段区间最长的合法前驱(h_),最长合法后驱(t_),一段中最长的合法区间 ...
- bzoj4543[POI2014]Hotel
题目链接 bzoj4543 [POI2014]Hotel 题解 这不是裸地点分嘛 ,我真傻,真的 n^2 这不是是sb题,~滑稽 ~ 枚举点转换为无根树,暴力子树中点的深度 计数转移 令a b c d ...
- 浅谈线段树 (例题:[USACO08FEB]酒店Hotel)By cellur925
今天我们说说线段树. 我个人还是非常欣赏这种数据结构的.(逃)因为它足够优美,有递归结构,有左子树和右子树,还有二分的思想. emm这个文章打算自用,就不写那些基本的操作了... 1° 简单的懒标记( ...
- 洛谷P2894 [USACO08FEB]酒店Hotel
P2894 [USACO08FEB]酒店Hotel https://www.luogu.org/problem/show?pid=2894 题目描述 The cows are journeying n ...
- P2894 [USACO08FEB]酒店Hotel
P2894 [USACO08FEB]酒店Hotel 简单的线段树维护区间信息. 维护三个值,一个是从左端点能拓展的长度,一个是从右端点能脱产的的长度.另一个是整个区间内的最大连续零一长度. 记录这三个 ...
- 洛谷 P2894 [USACO08FEB]酒店Hotel 解题报告
P2894 [USACO08FEB]酒店Hotel 题目描述 The cows are journeying north to Thunder Bay in Canada to gain cultur ...
- 线段树【洛谷P2894】 [USACO08FEB]酒店Hotel
P2894 [USACO08FEB]酒店Hotel 参考样例,第一行输入n,m ,n代表有n个房间,编号为1---n,开始都为空房,m表示以下有m行操作,以下 每行先输入一个数 i ,表示一种操作: ...
随机推荐
- 【HANA系列】SAP HANA STUDIO客户端升级更新
公众号:SAP Technical 本文作者:matinal 原文出处:http://www.cnblogs.com/SAPmatinal/ 原文链接:[HANA系列]SAP HANA STUDIO客 ...
- unittest中的断言方法
方法 用途 assertEqual(a,b) a=b assertNotEqual(a,b) a!=b assertTrue(x) x为True assertFa ...
- MySQL8 clone plugin
安装MySQl8.0.17 下载 MySQL8.0.17 二进制版本 https://dev.mysql.com/downloads/mysql/ 解压,修改权限 -linux-glibc2.-x86 ...
- ajax的contentType和dataType
1.contentType,明确告诉服务器我的请求的编码类型是json(严格来说是application/json),不设置的话,有默认值application/x-www-form-urlencod ...
- python学习第五十四天hashlib模块的使用
hash算法 hash也做散列,也称为哈希,主要用于信息安全领域中加密算法,hash就是找一种数据内容和数据存放地址直接的映射关系. md5算法 md5讯息算法,广泛使用密码函数 md5算法的特点 1 ...
- Distributed Deep Learning
安利一下刘铁岩老师的<分布式机器学习>这本书 以及一个大神的blog: https://zhuanlan.zhihu.com/p/29032307 https://zhuanlan.zhi ...
- Windows 10 IoT Core Dashboard 无法安装的问题
有人在answers.microsoft.com问这个问题,官方给了个这样的回答,然后还锁定了问题不让别人回复 您好, 了解到您在使用时遇到问题. 请您详细描述下您的操作,请问您是在打开安装程序还是在 ...
- 为ASP.NET按钮(Button)添加确认对话框
http://www.cnblogs.com/blodfox777/articles/1261303.html Button有两个点击事件: onclick 触发服务端事件,脚本为c#或VB.NET ...
- 免费申请 QQ 免费靓号
打开网址:https://ssl.zc.qq.com/v3/index-chs.html?type=3 说明:靓号一般分为 9位靓号 or 带有寓意的号码 激活规则:获取的"靓号" ...
- nmblookup - 基于TCP/IP上的NetBIOS客户用于查询NetBIOS名字的程序
总览 SYNOPSIS nmblookup [-M] [-R] [-S] [-r] [-A] [-h] [-B <broadcast address>] [-U <unicast a ...