[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 ,表示一种操作: ...
随机推荐
- 【转载】linux下如何使用sftp命令
原文地址:http://www.cnblogs.com/chen1987lei/archive/2010/11/26/1888391.html sftp 是一个交互式文件传输程式.它类似于 ftp, ...
- bzoj3028食物 关于(1+x+x2+x3+x4+...)^k的第i项系数就是c(i+k−1,k−1)的证明
关于(1+x+x2+x3+x4+...)^k的第i项系数就是c(i+k−1,k−1)的证明对于第i项,假设为5x^5=x^0*x^5x^5=x^1*x^4x^5=x^2*x^3........也就是说 ...
- Win10.输入法(控制面板)
1.之前 Win7 都是每个进程都是自己的输入法. 但是到了Win10 默认情况下 输入法是全局的,输入法切换成中文 所有进程都变成 中文输入,又是很不方便 也不习惯... 2.感觉 WIn10 真不 ...
- spring -boot定时任务 quartz 基于 JobDetailFactoryBean实现
这个有点小问题 尚未解决 后期优化 基于 JobDetailFactoryBean实现 依赖包 <dependencies> <dependency> <groupId ...
- [转帖]Intel Xeon路线图:7nm处理器要上DDR5、PCIe 5.0
Intel Xeon路线图:7nm处理器要上DDR5.PCIe 5.0 https://www.cnbeta.com/articles/tech/849631.htm 在月初的投资者会议上,Intel ...
- [转帖]16nm国内最先进 兆芯展示x86 KX-6000八核处理器
16nm国内最先进 兆芯展示x86 KX-6000八核处理器 https://www.cnbeta.com/articles/tech/847125.htm 在近日的2019北京国际互联网科技博览会暨 ...
- Spring中的常见注解
@Controller 标识一个该类是Spring MVC controller处理器,用来创建处理http请求的对象. @RestController Spring4之后加入的注解,原来在@Co ...
- ccs中a链接的四种状态
什么是超链接? 超链接通俗地指从一个网页指向一个目标的连接关系,这个目标可以是另一个网页,也可以是相同网页上的不同位置,还可以是一个图片,一个电子邮件地址,一个文件,甚至是一个应用程序.而在一个网页中 ...
- HDUSTOJ-1559 Vive la Difference!(简单题)
1559: Vive la Difference! 时间限制: 3 Sec 内存限制: 128 MB提交: 18 解决: 14[提交][状态][讨论版] 题目描述 Take any four po ...
- PY个树状数组
树状数组看起来比较简单,于是就挑它下手了... 于是生活终于也对咱下手了... 要讲的就两个东西,一个是开数组,全局变量写最前面,数组是这么开的: f=[0 for i in range(500005 ...