正确的题解

首先我们都知道这题要用线段树做。考虑维护靠左边的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 题解的更多相关文章

  1. USACO08FEB Hotel

    题目传送门 线段树维护区间 线段树结构体 struct zzz{ int l,r,mi; //l为以左端点的为起点的最长子串 //r为以右端点为终点的最长子串 //mi是区间内部的最长子串 }tree ...

  2. POJ3667 Hotel 题解

    和最大子段和的思路是一样的,可以记 \(lmax,rmax,dat\) 分别表示从当前区间最靠左/右的最大连续空子段和当前区间的最大连续空子段. 需要用延迟标记,每次遇到开房操作先ask,如果能找到就 ...

  3. 线段树||BZOJ1593: [Usaco2008 Feb]Hotel 旅馆||Luogu P2894 [USACO08FEB]酒店Hotel

    题面:P2894 [USACO08FEB]酒店Hotel 题解:和基础的线段树操作差别不是很大,就是在传统的线段树基础上多维护一段区间最长的合法前驱(h_),最长合法后驱(t_),一段中最长的合法区间 ...

  4. bzoj4543[POI2014]Hotel

    题目链接 bzoj4543 [POI2014]Hotel 题解 这不是裸地点分嘛 ,我真傻,真的 n^2 这不是是sb题,~滑稽 ~ 枚举点转换为无根树,暴力子树中点的深度 计数转移 令a b c d ...

  5. 浅谈线段树 (例题:[USACO08FEB]酒店Hotel)By cellur925

    今天我们说说线段树. 我个人还是非常欣赏这种数据结构的.(逃)因为它足够优美,有递归结构,有左子树和右子树,还有二分的思想. emm这个文章打算自用,就不写那些基本的操作了... 1° 简单的懒标记( ...

  6. 洛谷P2894 [USACO08FEB]酒店Hotel

    P2894 [USACO08FEB]酒店Hotel https://www.luogu.org/problem/show?pid=2894 题目描述 The cows are journeying n ...

  7. P2894 [USACO08FEB]酒店Hotel

    P2894 [USACO08FEB]酒店Hotel 简单的线段树维护区间信息. 维护三个值,一个是从左端点能拓展的长度,一个是从右端点能脱产的的长度.另一个是整个区间内的最大连续零一长度. 记录这三个 ...

  8. 洛谷 P2894 [USACO08FEB]酒店Hotel 解题报告

    P2894 [USACO08FEB]酒店Hotel 题目描述 The cows are journeying north to Thunder Bay in Canada to gain cultur ...

  9. 线段树【洛谷P2894】 [USACO08FEB]酒店Hotel

    P2894 [USACO08FEB]酒店Hotel 参考样例,第一行输入n,m ,n代表有n个房间,编号为1---n,开始都为空房,m表示以下有m行操作,以下 每行先输入一个数 i ,表示一种操作: ...

随机推荐

  1. 【HANA系列】SAP HANA STUDIO客户端升级更新

    公众号:SAP Technical 本文作者:matinal 原文出处:http://www.cnblogs.com/SAPmatinal/ 原文链接:[HANA系列]SAP HANA STUDIO客 ...

  2. unittest中的断言方法

    方法        用途 assertEqual(a,b)      a=b assertNotEqual(a,b)    a!=b assertTrue(x)     x为True assertFa ...

  3. MySQL8 clone plugin

    安装MySQl8.0.17 下载 MySQL8.0.17 二进制版本 https://dev.mysql.com/downloads/mysql/ 解压,修改权限 -linux-glibc2.-x86 ...

  4. ajax的contentType和dataType

    1.contentType,明确告诉服务器我的请求的编码类型是json(严格来说是application/json),不设置的话,有默认值application/x-www-form-urlencod ...

  5. python学习第五十四天hashlib模块的使用

    hash算法 hash也做散列,也称为哈希,主要用于信息安全领域中加密算法,hash就是找一种数据内容和数据存放地址直接的映射关系. md5算法 md5讯息算法,广泛使用密码函数 md5算法的特点 1 ...

  6. Distributed Deep Learning

    安利一下刘铁岩老师的<分布式机器学习>这本书 以及一个大神的blog: https://zhuanlan.zhihu.com/p/29032307 https://zhuanlan.zhi ...

  7. Windows 10 IoT Core Dashboard 无法安装的问题

    有人在answers.microsoft.com问这个问题,官方给了个这样的回答,然后还锁定了问题不让别人回复 您好, 了解到您在使用时遇到问题. 请您详细描述下您的操作,请问您是在打开安装程序还是在 ...

  8. 为ASP.NET按钮(Button)添加确认对话框

    http://www.cnblogs.com/blodfox777/articles/1261303.html Button有两个点击事件: onclick 触发服务端事件,脚本为c#或VB.NET ...

  9. 免费申请 QQ 免费靓号

    打开网址:https://ssl.zc.qq.com/v3/index-chs.html?type=3 说明:靓号一般分为 9位靓号 or 带有寓意的号码 激活规则:获取的"靓号" ...

  10. nmblookup - 基于TCP/IP上的NetBIOS客户用于查询NetBIOS名字的程序

    总览 SYNOPSIS nmblookup [-M] [-R] [-S] [-r] [-A] [-h] [-B <broadcast address>] [-U <unicast a ...