Tunnel Warfare 线段树 区间合并|最大最小值
B - Tunnel WarfareHDU - 1540
这个有两种方法,一个是区间和并,这个我个人感觉异常恶心
第二种方法就是找最大最小值 kuangbin——线段树专题 H - Tunnel Warfare <-- 这个博客讲的很清楚
#include <cstdio>//法一:最大值最小值法,这个就是每一个点,如果这个点没有被摧毁,那就找到这个点最左边的和最右边的
#include <cstdlib>//最大-最小+1.这个就是这个最大连续长度。
#include <queue>//建树,很简单,主要就是query和update。
#include <algorithm>//这个地方的怎么去找一个包含一个数的一个区间的最大最小值呢?
#include <vector>//这个就是从上面往下面查询的过程中,就去找,如果是找最大值就去max,最小值就取min
#include <cstring>//这个要注意建树,这个区间的最大值的意思是,小于等于这个数的最大的被炸了的村庄,这个就说明,开始最大值为0,因为没有任何一个村庄被炸
#include <string>//区间的最小值,意思是大于等于这个数,被炸了的村庄的最小值,开始为n+1.因为没有村庄被炸。
#include <iostream>
#include <stack>
#define inf 0x3f3f3f3f
using namespace std;
const int maxn = 1e5 + ;
struct node
{
int l, r;
int mx, mn;
}tree[maxn*];
int n;
void build(int id,int l,int r)
{
tree[id].l = l;
tree[id].r = r;
if(l==r)
{
tree[id].mn = n+;
tree[id].mx = ;
return;
}
int mid = (l + r) >> ;
build(id << , l, mid);
build(id << | , mid + , r);
tree[id].mx = max(tree[id << ].mx, tree[id << | ].mx);
tree[id].mn = min(tree[id << ].mn, tree[id << | ].mn);
} void update_max(int id,int x,int z)
{
if(tree[id].l==tree[id].r)
{
tree[id].mx = z;
return;
}
int mid = (tree[id].l + tree[id].r) >> ;
if(x<=mid) update_max(id << , x, z);
if (x > mid) update_max(id << | , x, z);
tree[id].mx = max(tree[id << ].mx, tree[id << | ].mx);
} void update_min(int id,int x,int z)
{
if(tree[id].l==tree[id].r)
{
tree[id].mn = z;
return;
}
int mid = (tree[id].l + tree[id].r) >> ;
if (x <= mid) update_min(id << , x, z);
if (x > mid) update_min(id << | , x, z);
tree[id].mn = min(tree[id << ].mn, tree[id << | ].mn);
} int query_max(int id,int x,int y)
{
int ans = ;
if(x<=tree[id].l&&y>=tree[id].r)
{
return tree[id].mx;
}
int mid = (tree[id].l + tree[id].r) >> ;
if (x <= mid) ans = max(ans, query_max(id << , x, y));
if (y > mid) ans = max(ans, query_max(id << | , x, y));
return ans;
} int query_min(int id,int x,int y)
{
int ans = inf;
if(x<=tree[id].l&&y>=tree[id].r)
{
return tree[id].mn;
}
int mid = (tree[id].l + tree[id].r) >> ;
if (x <= mid) ans = min(ans, query_min(id << , x, y));
if (y > mid) ans = min(ans, query_min(id << | , x, y));
return ans;
} int main()
{ int m, x;
while(cin >> n >> m)
{
stack<int>sta;
build(, , n);
while(m--)
{
char s[];
scanf("%s", s);
if(s[]=='D')
{
cin >> x;
update_max(, x, x);
update_min(, x, x);
sta.push(x);
}
if(s[]=='R')
{
int y = sta.top(); sta.pop();
update_max(, y, );
update_min(, y, n + );
}
if(s[]=='Q')
{
cin >> x;
int L = query_min(, x, n + );
int R = query_max(, , x);
//printf("%d %d\n", L, R);
if (L == R) printf("0\n");
else printf("%d\n", L - R - );
}
}
} return ;
}
方法一
//法二:区间合并,这个应该更好懂一点,就是维护一下一个区间的前缀后缀长度
//这个更新应该比较简单,
#include <cstdio>
#include <cstdlib>
#include <queue>
#include <algorithm>
#include <vector>
#include <cstring>
#include <string>
#include <iostream>
#include <stack>
#define inf 0x3f3f3f3f
using namespace std;
const int maxn = 1e5 + ;
struct node
{
int l, r, len;
int max_pre, max_last;
}tree[maxn*]; void push_up(int id)
{
tree[id].max_pre = tree[id << ].max_pre;
tree[id].max_last = tree[id << | ].max_last;
if (tree[id << ].max_pre == tree[id << ].len)
{
tree[id].max_pre += tree[id << | ].max_pre;
}
if(tree[id<<|].max_last==tree[id<<|].len)
{
tree[id].max_last += tree[id << ].max_last;
}
} void build(int id,int l,int r)
{
tree[id].l = l;
tree[id].r = r;
tree[id].len = r - l + ;
if(l==r)
{
tree[id].max_pre = tree[id].max_last = ;
return;
}
int mid = (l + r) >> ;
build(id << , l, mid);
build(id << | , mid + , r);
push_up(id);
} void update(int id,int x,int z)
{
if(tree[id].l==tree[id].r)
{
tree[id].max_pre = z;
tree[id].max_last = z;
return;
}
int mid = (tree[id].l + tree[id].r) >> ;
if (x <= mid) update(id << , x, z);
else update(id << | , x, z);
push_up(id);
} int query_pre(int id,int x,int y)
{
int ans = , res = ;
if(x<=tree[id].l&&y>=tree[id].r)
{
//printf("tree[%d].max_pre=%d\n", id, tree[id].max_pre);
return tree[id].max_pre;
}
//printf("id=%d x=%d y=%d\n", id, x, y);
int mid = (tree[id].l + tree[id].r) >> ;
if (x <= mid) ans=query_pre(id << , x, y);
if (y > mid) res=query_pre(id << | , x, y);
//printf("id=%d ans=%d res=%d mid=%d\n",id, ans, res,mid);
if (ans >= mid - x + )
{
//printf("tree[%d].max_pre=%d mid=%d x=%d\n",id, tree[id].max_pre, mid, x);
ans += res;
}
return ans;
} int query_last(int id,int x,int y)
{
int ans = , res = ;
if (x <= tree[id].l&&y >= tree[id].r)
{
//printf("tree[%d].last=%d\n", id, tree[id].max_last);
return tree[id].max_last;
}
//printf("id=%d x=%d y=%d\n", id, x, y);
int mid = (tree[id].l + tree[id].r) >> ;
if (x <= mid) ans = query_last(id << , x, y);
if (y > mid) res = query_last(id << | , x, y);
//printf("id=%d mid=%d ans=%d res=%d\n", id, mid,ans, res);
if (res >= y-mid)
{
//printf("tree[%d].max_last=%d mid=%d x=%d\n",id,tree[id].max_last, mid, x);
res += ans;
}
return res;
} int main()
{
int n, m;
while(scanf("%d %d",&n,&m)!=EOF)
{
stack<int>sta;
build(, , n);
while(m--)
{
char s[];
scanf("%s", s);
if(s[]=='D')
{
int x;
cin >> x;
update(, x, );
sta.push(x);
}
if(s[]=='R')
{
int y = sta.top(); sta.pop();
update(, y, );
}
if(s[]=='Q')
{
int x;
cin >> x;
int ans = query_pre(, x, n);
ans += query_last(, , x);
if(ans) printf("%d\n", ans-);
else printf("0\n");
}
}
}
return ;
}
/*
7 10
D 3
D 6
D 5
Q 4
Q 5
R
Q 4
R
Q 4
Q 3
*/
方法二
Tunnel Warfare 线段树 区间合并|最大最小值的更多相关文章
- HDU 1540 Tunnel Warfare 线段树区间合并
Tunnel Warfare 题意:D代表破坏村庄,R代表修复最后被破坏的那个村庄,Q代表询问包括x在内的最大连续区间是多少 思路:一个节点的最大连续区间由(左儿子的最大的连续区间,右儿子的最大连续区 ...
- HDU1540 Tunnel Warfare —— 线段树 区间合并
题目链接:https://vjudge.net/problem/HDU-1540 uring the War of Resistance Against Japan, tunnel warfare w ...
- hdu 1540 Tunnel Warfare 线段树 区间合并
题意: 三个操作符 D x:摧毁第x个隧道 R x:修复上一个被摧毁的隧道,将摧毁的隧道入栈,修复就出栈 Q x:查询x所在的最长未摧毁隧道的区间长度. 1.如果当前区间全是未摧毁隧道,返回长度 2. ...
- Tunnel Warfare HDU 1540 区间合并+最大最小值
Tunnel Warfare HDU 1540 区间合并+最大最小值 题意 D x是破坏这个点,Q x是表示查询以x所在的最长的连续的点的个数,R是恢复上一次破坏的点. 题解思路 参考的大佬博客 这里 ...
- hdu 1540 Tunnel Warfare(线段树区间统计)
Tunnel Warfare Time Limit: 4000/2000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others) T ...
- Tunnel Warfare(HDU1540+线段树+区间合并)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1540 题目: 题意:总共有n个村庄,有q次操作,每次操作分为摧毁一座村庄,修复一座村庄,和查询与询问的 ...
- 线段树区间合并优化dp——cf1197E(好)
线段树优化dp的常见套路题,就是先按某个参数排序,然后按这个下标建立线段树,再去优化dp 本题由于要维护两个数据:最小值和对应的方案数,所以用线段树区间合并 /* dp[i]表示第i个套娃作为最内层的 ...
- POJ 3667 Hotel(线段树 区间合并)
Hotel 转载自:http://www.cnblogs.com/scau20110726/archive/2013/05/07/3065418.html [题目链接]Hotel [题目类型]线段树 ...
- HDU 3911 线段树区间合并、异或取反操作
题目:http://acm.hdu.edu.cn/showproblem.php?pid=3911 线段树区间合并的题目,解释一下代码中声明数组的作用: m1是区间内连续1的最长长度,m0是区间内连续 ...
随机推荐
- git获取特定的commit
git reset --hard [commit_id]
- java 代码执行cmd 返回值异常 (关于JAVA Project.waitfor()返回值是1)
关于JAVA Project.waitfor()返回值是1 0条评论 Project.waitfor()返回值是1,找了很久从网上没有发现关于1的说明. 这时对源代码调试了一下,发现Project ...
- 玩家的numpertpry 对象 中 不仅仅要同步 君主武将的等级,阶级也要同步
因为好多列表 中 需要 批量查询 玩家的等级 和阶级(用来显示玩家icon颜色用的),如果阶级 在numperty 中已同步 的话,就不用批量去查玩家武将列表了.同理如果其他属性也经常用的话也可以同步 ...
- Java面向对象的总结
面向对象的程序设计 1.面向对象 核心:以类的方式组织代码,以对象的方式封装数据 比喻:也就是说类是没有数据的,给了数据之后的类就是对象 封装 继承 多态 2.方法 a.一个方法只有一个返回值,只有一 ...
- 第十节:xml、re、logging模块
XML模块:(用到的时候再看)tree=xml.parse('xmltest.xml')root= tree.getroot()print(root.tag) 打印对象的标签root.attrib 获 ...
- ModuleNotFoundError: No module named 'sklearn.cross_validation'
本文为CSDN博主「不服输的南瓜」的原创文章,遵循 CC 4.0 BY-SA 版权协议 原文链接 ModuleNotFoundError: No module named 'sklearn.cross ...
- ASE课程总结 by 朱玉影
收获: 最大的收获应该就是对待选题要慎重吧,虽然前期做了一下调研,但是还是不够,所以到最后我们的项目才会不能公开发布,项目中间也是波折不断,导致我们走了很多弯路,浪费了很多时间吧.选题一定要慎重,慎重 ...
- stand up meeting 12/3/2015
part 组员 今日工作 工作耗时/h 明日计划 工作耗时/h UI 冯晓云 初始化弹窗的弹出位置并捕捉弹窗区域内的鼠标控制事件,初步解决弹窗的拖拽功能: 6 UWP对控件的支持各种看不懂,属性 ...
- 数据结构与算法--树(tree)结构
树 二叉树 遍历原则:前序遍历是根左右, 中序遍历是左根右,后序遍历是左右根. 二叉搜索树 特点:对于树中的每个节点X,它的左子树中所有节点的值都小于X,右子树中所有节点的值都大于X. 遍历:采取二叉 ...
- 详解 Hashtable
至于HashTable,本人只想说,除了它们各自的特点是截然相反外,其余性质 以及 用法和HashMap的性质几乎一样, (有关Map集合的基本性质,请观看本人博文-- <详解 Map集合> ...