Tunnel Warfare

                                                            Time Limit: 4000/2000 MS (Java/Others)    Memory Limit: 65536/32768
K (Java/Others)

                                                                                                                       链接:hdu 1540        POJ
2892

 Problem Description 
During the War of Resistance Against Japan, tunnel warfare was carried out extensively in the vast areas of north China Plain. Generally speaking, villages connected by tunnels lay in a line. Except the two at the ends, every village was directly connected
with two neighboring ones.

Frequently the invaders launched attack on some of the villages and destroyed the parts of tunnels in them. The Eighth Route Army commanders requested the latest connection state of the tunnels and villages. If some villages are severely isolated, restoration
of connection must be done immediately!
Input
The first line of the input contains two positive integers n and m (n, m ≤ 50,000) indicating the number of villages and events. Each of the next m lines describes an event.



There are three different events described in different format shown below:



D x: The x-th village was destroyed.



Q x: The Army commands requested the number of villages that x-th village was directly or indirectly connected with including itself.



R: The village destroyed last was rebuilt.
Output
Output the answer to each of the Army commanders’ request in order on a separate line.
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

题意:

给定N个点。点的编号是从1 ~ N,M次‘Q’或‘D或‘R’的操作,“D  x”表示的是破坏这个点x,“R”是修复之前破坏的点,“Q x”表示询问点x所在连续区间的长度。假定最初每一个点都是好的。

分析:

首先对操作进行分析:能够把D操作看成把区间在点x出截断,R操作是把x左右【包含点x】进行合并。

每次合并两个区间 [a,b] , [c,d] 的时候,得到的新的区间[a,d] 的最大连续区间长度为:

Len(a,d)  =  max{ Len(a,b),Len(c,d) ,End(b) + Begin(c) };  <==有一点分治的思想

End(b)表示以b结尾的最大连续区间长度。Begin(c)表示以c开头的最大连续区间长度,这里我们能够知道,合并两个区间,并非简单的区间加减。而是还要保持一个以区间第一个元素開始的最大连续区间长度以及 以区间最后一个元素结尾的最大连续区间长度。

那么。如今问题就比較好办了。线段树每一个节点包括三个信息。各自是以区间第一个元素開始的最大连续区间长度ln。以区间最后一个元素结尾的最大连续区间长度rn,以及区间最大连续长度mn。

接下来,便是Q操作了,询问点x 所在连续区间的长度。

在递归左儿子节点的时候,注意一下,x是否在以左儿子区间最后一个元素结尾长度为左儿子rn的范围里面,假设在,那么x所在的连续区间便有可能包括右儿子的一部分,递归右儿子的时候,同理!

/****************************>>>>HEADFILES<<<<****************************/
#include <set>
#include <map>
#include <list>
#include <cmath>
#include <queue>
#include <vector>
#include <cstdio>
#include <string>
#include <cstring>
#include <iomanip>
#include <iostream>
#include <sstream>
#include <algorithm>
using namespace std;
/****************************>>>>>DEFINE<<<<<*****************************/
#define fst first
#define snd second
#define root 1,N,1
#define lson l,mid,rt<<1
#define rson mid+1,r,rt<<1|1
#define PB(a) push_back(a)
#define MP(a,b) make_pair(a,b)
#define CASE(T) for(scanf("%d",&T);T--;)
#define FIN freopen("input.txt","r",stdin)
#define FOUT freopen("output.txt","w",stdout)
//#pragma comment(linker, "/STACK:1024000000,1024000000")
typedef __int64 LL;
const int INF = 0x3f3f3f3f;
/****************************>>>>SEPARATOR<<<<****************************/
const int maxn = 50000 + 5;
int N, M;
struct Node
{
int ln, rn, mn;
} segtree[maxn << 4];
inline void PushUp(const int& rt, const int& l, const int& r)
{
segtree[rt].ln = segtree[rt << 1].ln;
segtree[rt].rn = segtree[rt << 1 | 1].rn;
segtree[rt].mn = max(segtree[rt << 1].rn + segtree[rt << 1 | 1].ln,
max(segtree[rt << 1].mn, segtree[rt << 1 | 1].mn));
int mid = (l + r) >> 1;
if(segtree[rt << 1].mn == mid - l + 1) segtree[rt].ln += segtree[rt << 1 | 1].ln;
if(segtree[rt << 1 | 1].mn == r - (mid + 1) + 1) segtree[rt].rn += segtree[rt << 1].rn;
}
void Build(int l, int r, int rt)
{
segtree[rt].ln = segtree[rt].rn = segtree[rt].mn = r - l + 1;
if(l == r)
{
return ;
}
int mid = (l + r) >> 1;
Build(lson);
Build(rson);
}
void Update(int pos, int val, int l, int r, int rt)
{
if(l == r)
{
segtree[rt].ln = segtree[rt].rn = segtree[rt].mn = val;
return;
}
int mid = (l + r) >> 1;
if(pos <= mid)
Update(pos, val, lson);
else
Update(pos, val, rson);
PushUp(rt, l, r);
}
int Query(int pos, int l, int r, int rt)
{
if(r - l + 1 == segtree[rt].mn || r == l || segtree[rt].mn == 0)
{
return segtree[rt].mn;
}
int mid = (l + r) >> 1, ret = 0;
if(pos <= mid)
{
if(pos >= mid - segtree[rt << 1].rn + 1)
ret = Query(pos, lson) + Query(mid + 1, mid + 1, r, rt << 1 | 1);
else ret = Query(pos, lson);
}
else
{
if(pos <= (mid + 1) + segtree[rt << 1 | 1].ln - 1)
ret = Query(pos, rson) + Query(mid, l, mid, rt << 1);
else ret = Query(pos, rson);
}
return ret;
}
int destroy[maxn];
int main()
{
//FIN;
while(~scanf("%d %d", &N, &M))
{
Build(root);
char Op[5];
int x, cnt = 0;
for(int i = 0; i < M; i++)
{
scanf("%s", Op);
if(Op[0] == 'Q')
{
scanf("%d", &x);
printf("%d\n", Query(x, root));
}
else if(Op[0] == 'D')
{
scanf("%d", &x);
Update(x, 0, root);
destroy[cnt++] = x;
}
else
{
x = destroy[--cnt];
Update(x, 1, root);
}
}
}
}

一年之后,再写了一遍这个题目。感觉代码风格还是变化了一点。再贴个代码~

#include <bits/stdc++.h>

using namespace std;

const int MX = 5e4 + 5;

#define lch     (rt << 1)
#define rch (rt << 1 | 1) typedef pair<int, int> PII; struct Seg {
/**
* status:
* 0 -- Bad
* 1 -- Good
* -1 -- not cover
* I: 节点相应区间两端点以及两短点的标号
*/
int sum, status;
PII I[2];
} seg[MX * 3]; int N, M, x;
char oper[5]; inline void pushUp(int rt) {
if(seg[lch].status == seg[rch].status) seg[rt].status = seg[lch].status;
else seg[rt].status = -1; if(seg[lch].status == 1 && seg[rch].status == 1) seg[rt].sum = seg[lch].sum + seg[rch].sum;
else seg[rt].sum = 0;
} void build(int l, int r, int rt) {
int temp;
if(l == r) {
seg[rt].sum = seg[rt].status = 1;
seg[rt].I[0] = make_pair(l, rt);
seg[rt].I[1] = make_pair(r, rt);
return;
}
int mid = (l + r) >> 1;
build(l, mid, lch);
build(mid + 1, r, rch);
pushUp(rt);
seg[rt].I[0] = seg[lch].I[0];
seg[rt].I[1] = seg[rch].I[1];
} void update(int pos, int v, int l, int r, int rt) {
if(l == r) {
seg[rt].status = v;
seg[rt].sum = (v == 1) ? 1 : 0;
return;
}
int mid = (l + r) >> 1;
if(pos <= mid) update(pos, v, l, mid, lch);
else update(pos, v, mid + 1, r, rch);
pushUp(rt);
} int query_status(int pos, int l, int r, int rt) {
if(l == r) return seg[rt].status;
int mid = (l + r) >> 1;
if(pos <= mid) return query_status(pos, l, mid, lch);
else return query_status(pos, mid + 1, r, rch);
} int query(int pos, int l, int r, int rt, int d) {
if(seg[rt].status == 0) return 0;
int mid = (l + r) >> 1, ret = 0;
if(seg[rt].status == 1) {
ret += seg[rt].sum;
PII &lb = seg[rt].I[0], &ub = seg[rt].I[1];
if(lb.first != 1 && seg[lb.second].status == 1 && d != 1)
ret += query(lb.first - 1, 1, N, 1, -1);
if(ub.first != N && seg[ub.second].status == 1 && d != -1)
ret += query(ub.first + 1, 1, N, 1, 1);
return ret;
}
if(pos <= mid) {
ret += query(pos, l, mid, lch, d);
} else {
ret += query(pos, mid + 1, r, rch, d);
}
return ret;
} int main() {
// freopen("input.txt", "r", stdin);
while(~scanf("%d %d", &N, &M)) {
build(1, N, 1);
stack<int> buf;
while(M --) {
scanf("%s", oper);
if(oper[0] == 'D') {
scanf("%d", &x); buf.push(x);
update(x, 0, 1, N, 1);
} else if(oper[0] == 'Q') {
scanf("%d", &x);
int k = query_status(x, 1, N, 1), ans = 0;
if(k != 0) {
ans = query(x, 1, N, 1, 0);
}
printf("%d\n", ans);
} else {
if(buf.empty()) continue;
x = buf.top(); buf.pop();
update(x, 1, 1, N, 1);
}
}
}
return 0;
}

hdu 1540/POJ 2892 Tunnel Warfare 【线段树区间合并】的更多相关文章

  1. HDU 1540 POJ 2892 Tunnel Warfare

    线段树 区间合并 单点修改 区间查询.又是1秒钟构思,差错查了好久... ... 发现一个int型的定义成了char型,打脸. #include <stdio.h> #include &l ...

  2. HDU 1540 / POJ 2892 Tunnel Warfare (单点更新,区间合并,求包含某点的最大连续个数)

    题意:一条线上有n个点,D x是破坏这个点,Q x是表示查询x所在的最长的连续的点的个数,R是恢复上一次破坏的点. 思路:这题的关键是查询. 将被毁的村庄看成空位,当查询某个点的时候,如果我们知道它左 ...

  3. HDU 1540 Tunnel Warfare 线段树区间合并

    Tunnel Warfare 题意:D代表破坏村庄,R代表修复最后被破坏的那个村庄,Q代表询问包括x在内的最大连续区间是多少 思路:一个节点的最大连续区间由(左儿子的最大的连续区间,右儿子的最大连续区 ...

  4. Tunnel Warfare 线段树 区间合并|最大最小值

    B - Tunnel WarfareHDU - 1540 这个有两种方法,一个是区间和并,这个我个人感觉异常恶心 第二种方法就是找最大最小值 kuangbin——线段树专题 H - Tunnel Wa ...

  5. HDU1540 Tunnel Warfare —— 线段树 区间合并

    题目链接:https://vjudge.net/problem/HDU-1540 uring the War of Resistance Against Japan, tunnel warfare w ...

  6. hdu 1540 Tunnel Warfare 线段树 区间合并

    题意: 三个操作符 D x:摧毁第x个隧道 R x:修复上一个被摧毁的隧道,将摧毁的隧道入栈,修复就出栈 Q x:查询x所在的最长未摧毁隧道的区间长度. 1.如果当前区间全是未摧毁隧道,返回长度 2. ...

  7. hdu 3911 Black And White (线段树 区间合并)

    题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=3911 题意: 给你一段01序列,有两个操作: 1.区间异或,2.询问区间最长的连续的1得长度 思路: ...

  8. HDU 3911 Black And White(线段树区间合并+lazy操作)

    开始以为是水题,结果...... 给你一些只有两种颜色的石头,0为白色,1为黑色. 然后两个操作: 1 l r 将[ l , r ]内的颜色取反 0 l r 计算[ l , r ]内最长连续黑色石头的 ...

  9. Poj 3667——hotel——————【线段树区间合并】

    Hotel Time Limit: 3000MS   Memory Limit: 65536K Total Submissions: 13124   Accepted: 5664 Descriptio ...

随机推荐

  1. 图文详解前端CSS中的Grid布局,你真的可以5分钟掌握

    前言 网站的布局是一个网站设计的根本,CSS的Grid布局已经成为了未来网站布局的基本方式. 今天这篇文章我们通过图文,一起看看如何自己实现Grid布局方式. CSS 第一个Grid布局 首先我们看看 ...

  2. 牛客网 暑期ACM多校训练营(第二场)D.money-贪心 or 动态规划

    D.money 贪心,直接贴官方的题解吧. 题目大意 你要按照顺序依次经过n个商店,每到达一个商店你可以购买一件商品,也可以出售你手中的商品. 同一时刻你手上最多拿一件商品.在第i个商店购买和出售的代 ...

  3. UVA 1395 Slim Span 最小生成树

    题意: 给你一个图,让你求这个图中所有生成树中满足题目条件的,这个条件是生成树中最长边与最短边的差值最小. 思路: 根据最小瓶颈生成树的定义:在一个有权值的无向图中,求一个生成树最大边的权值尽量小.首 ...

  4. Python的程序结构[0] -> 属性/Property[0] -> 类属性、实例属性和私有属性

    类属性.实例属性和私有属性 Python中类的属性主要包括类属性,实例属性和私有属性,下面是对三种属性的简单介绍 类属性 / Class Property 类属性在__init__()之外初始化,在外 ...

  5. 山东多校联合模拟赛 Day1

    矩形计数(rect) Description 给出圆周上的 \(N\) 个点,请你计算出以这些点中的任意四个为四个角,能构成多少个矩 形. 点的坐标是这样描述的,给定一个数组 \(v[1..N]\), ...

  6. mysql里的知识

    1.mysql基础 (1)mysql存储结构:数据库->表-> 数据   sql语句 (2)管理数据库: 增加: create database 数据库 default character ...

  7. 深入JS正则先行断言

    这里是 Mastering Lookahead and Lookbehind 文章的简单翻译,这篇文章是在自己搜索问题的时候stackoverflow上回答问题的人推荐的,看完觉得写得很不错.这里的简 ...

  8. [COCI2015]TRAKTOR

    题目大意: 一个$X\times Y(X,Y\leq10^5)$的格子中,每秒钟依次$n(n\leq10^6)$个蘑菇, 告诉你每个蘑菇出现的时间和位置,问何时第一次出现$k(2\leq k\leq ...

  9. 图床plus演示 | 图床及在线分享演示文稿工具

    文章目录 关于图床 什么是图床? 墙内 墙外 关于在线分享演示文稿 在线分享演示文稿 工具分享 待补充 关于图床 什么是图床? 这并不是一个多么高大上的名词概念!用比较通俗的话来说,当你在撰写新文章时 ...

  10. Android APP打包时,出错:"XXX" is not translated in "af" (Afrikaans), "am" (Amharic), "ar" (Arabic).....

    "app_name" is not translated in "af" (Afrikaans), "am" (Amharic), &quo ...