Tunnel Warfare HDU - 1540 (线段树不同子树的合并)
2024-10-02 06:44:51
在抗日战争期间,华北平原广大地区进行了大规模的隧道战。 一般来说,通过隧道连接的村庄排成一列。 除了两端,每个村庄都与两个相邻的村庄直接相连。
入侵者经常对一些村庄发动袭击并摧毁其中的部分隧道。 八路军指挥官要求最新的隧道和村庄连接状态。 如果某些村庄严重隔离,必须立即恢复连接!
Input
输入的第一行包含两个正整数n和m(n,m≤50,000),表示村庄和事件的数量。 接下来的m行中的每一行描述一个事件。
以下所示的不同格式描述了三种不同的事件:
D x:第x个村庄被毁。
Q x:指挥官询问第x个村庄与其直接或间接相关的村庄数量。
R:最后毁坏的村庄被重建了。
Output
按顺序输出每个指挥官询问的答案。
Sample Input
7 9
D 3
D 6
D 5
Q 4
Q 5
R
Q 4
R
Q 4
Sample Output
1
0
2
4 分析:
维护每个节点所对应区间的左最长1串,右最长一串和区间最长1串。查询的时候如果当前区间最长1串长度为0或者满,直接返回0\满;如果x位于当前区间的“中部”(左孩子的右和右孩子的左连接而成),直接返回中部长度;除此以外就继续查找。
代码:
#include <bits/stdc++.h>//题目也没说多组输入啊,WA了好多次,想哭
using namespace std;
const int maxn = * 1e4 + ;
struct node
{
int l, r;
int ln, rn, mn;
}t[maxn << ]; int n, m; void pushup(int tar)
{
t[tar].ln = t[tar << ].ln, t[tar].rn = t[tar << | ].rn;
t[tar].mn = max(t[tar << ].mn, t[tar << | ].mn);
t[tar].mn = max(t[tar].mn, t[tar << ].rn + t[tar << | ].ln);
if (t[tar << ].ln == t[tar << ].r - t[tar << ].l + ) t[tar].ln = t[tar << ].ln + t[tar << | ].ln;
if (t[tar << | ].rn == t[tar << | ].r - t[tar << | ].l + ) t[tar].rn = t[tar << | ].rn + t[tar << ].rn;
} void build(int l, int r, int tar)
{
t[tar].l = l, t[tar].r = r;
t[tar].ln = t[tar].rn = t[tar].mn = r - l + ;
if (l == r) return;
int mid = (l + r) >> ;
build(l, mid, tar << );
build(mid + , r, tar << | );
} void update(int x, int state, int tar)
{
if (t[tar].l == t[tar].r)
{
t[tar].ln = t[tar].rn = t[tar].mn = state;
return;
}
int mid = (t[tar].l + t[tar].r) >> ;
if (x <= mid) update(x, state, tar << );
else if (x > mid) update(x, state, tar << | );
pushup(tar);
} int query(int x, int tar)//查询依据为,x必然存在于某个区间的中心部分,叶子节点除外,所以对叶子节点特判。
{
if (t[tar].l == t[tar].r) return 0;//如果是叶子节点直接输出0,说明 int mid = (t[tar].l + t[tar].r) >> ;
if (x >= t[tar << ].r - t[tar << ].rn + && x <= t[tar << | ].l + t[tar << | ].ln - ) return t[tar << ].rn + t[tar << | ].ln;
else if (x <= mid) return query(x, tar << );
else return query(x, tar << | );
} int main()
{
int n, m; while (cin >> n >> m)
{
stack<int> s;
char ope[];
int x; build(, n, );
while (m--)
{
cin >> ope;
if (ope[] == 'D')
{
cin >> x;
s.push(x);
update(x, , );
}
else if (ope[] == 'R')
{
x = s.top();
s.pop();
update(x, , );
}
else
{
cin >> x;
cout << query(x, ) << endl;
}
}
}
}
最新文章
- MYSQL实现主从复制
- Linux下取代top的进程管理工具 htop
- Bootstrap学习应用
- 关于Oracle GoldenGate中Extract的checkpoint的理解 转载
- Leetcode 382. Linked List Random Node
- BZOJ2149 : 拆迁队
- RDS MySQL 连接数满情况的处理
- C# Excel导入
- rdesktop remember
- hdu----(1950)Bridging signals(最长递增子序列 (LIS) )
- C#局域网桌面共享软件制作(一)
- jQuery1.9.1--queue队列源码分析(非动画部分)
- ubuntu搭建svn、git遇到的问题及解决办法
- MVC折线图应用
- 【cogs 775】山海经 ——Segment Tree
- 通用网页调用本地应用程序方案(windows平台)
- 信步漫谈之Jenkins&mdash;集成环境搭建
- git 常用命令思维导图
- C程序第三次作业
- 训练题(代码未检验)(序列前k大和问题)
热门文章
- 20152016-acmicpc-neerc-northern-subregional-contest J:Journey to the ";The World&#39;s Start";(单调队列+DP+二分)
- JSON.stringify() 的深入理解
- 判断List中是否含有某个实体bean
- Bzoj 3874: [Ahoi2014&;Jsoi2014]宅男计划 三分+贪心
- C#类型详解
- markdown浅谈
- 个人永久性免费-Excel催化剂功能第83波-遍历文件夹内文件信息特别是图像、音视频等特有信息
- 万能RecyclerView的数据适配器BaseRecyclerViewAdapterHelper
- [sublime3] 在linux下的终端中使用sublime3打开文件
- MySql的数据库优化到底优化啥了都(3)