【BZOJ-2325】道馆之战 树链剖分 + 线段树
2325: [ZJOI2011]道馆之战
Time Limit: 40 Sec Memory Limit: 256 MB
Submit: 1153 Solved: 421
[Submit][Status][Discuss]
Description

Input
Output
包含若干行,每行一个整数。即对于输入中的每个询问,依次输出一个答案。
Sample Input
1 2
2 3
2 4
1 5
.#
..
#.
.#
..
Q 5 3
C 1 ##
Q 4 5
Sample Output
3
HINT
Source
Solution
树链剖分+线段树维护(类似)连通性。 和 堵塞的交通 维护方法类似。但是并没有做过。
题目大意:每个节点分为ab连个块,这些节点之间联通呈树形,每个节点的ab可能为障碍或者是空地,从一个节点上的a可以到这个节点的b,或者和这个节点联通的所有a(前提是路径上无障碍)。询问节点x到节点y的方向上最多走多少步,支持单点修改.
这样如果是链上的情况,可以等价为一个2*N的矩阵,上面一行表示所有的a,下面一行表示所有的b,等效到树上,就是多个2*1个矩阵连成的树。
维护方法就是,对于一个区间,维护8个量分别表示:
dis[0][0]表示从区间左端点上到右端点上,dis[0][1]表示从区间左端点下到右端点上,dis[1][0]表示从区间左端点上到右端点下,dis[1][1]表示从区间左端点下到右端点下
Dis[0][0]表示从区间左端点上最多移动数,Dis[0][1]表示从区间左端点下最多移动数,Dis[1][0]表示从区间右端点上最多移动数,Dis[1][1]表示从区间右端点下最多移动数
合并的时候就是讨论一下,画画图就可以得到。
有一个问题,在询问路径x-->y时,有一边到LCA的路径是从下到上的,所以在和另一边路径合并答案时要先反向,然后就可以了。
写的时候要注意细节,这个在Codeing的时候由于思路清晰,写的还是十分美观的,就是断断续续自己 想+写+调 搞了近两个小时...(话说题意真是不清晰)
Code
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<cmath>
using namespace std;
inline int read()
{
int x=; char ch=getchar();
while (ch<'' || ch>'') ch=getchar();
while (ch>='' && ch<='') x=x*+ch-'',ch=getchar();
return x;
}
#define MAXN 30010
#define INF 100000000
int N,M,id[MAXN]; char mp[MAXN][];
namespace SegmentTree
{
struct SgtNode
{
int l,r,dis[][],Dis[][];
SgtNode() { l=,r=; memset(dis,,sizeof(dis)); memset(Dis,,sizeof(Dis)); }
inline bool have() {return !dis[][] && !dis[][] && !dis[][] && !dis[][] && !Dis[][] && !Dis[][] && !Dis[][] && !Dis[][];}
inline void rever() { swap(dis[][],dis[][]),swap(Dis[][],Dis[][]),swap(Dis[][],Dis[][]); }
inline void modify(int a,int b)
{
dis[][]=a? :-INF; Dis[][]=Dis[][]=a;
dis[][]=b? :-INF; Dis[][]=Dis[][]=b;
if (a && b) Dis[][]=Dis[][]=Dis[][]=Dis[][]=dis[][]=dis[][]=;
else dis[][]=dis[][]=-INF;
}//单点修改时的赋值
}tree[MAXN<<];
// dis[0][0]表示从区间左端点上到右端点上,dis[0][1]表示从区间左端点下到右端点上,dis[1][0]表示从区间左端点上到右端点下,dis[1][1]表示从区间左端点下到右端点下
// Dis[0][0]表示从区间左端点上最多移动数,Dis[0][1]表示从区间左端点下最多移动数,Dis[1][0]表示从区间右端点上最多移动数,Dis[1][1]表示从区间右端点下最多移动数
#define ls now<<1
#define rs now<<1|1
inline void update(SgtNode &rt,SgtNode lson,SgtNode rson)
{
SgtNode tmp=rt;
if (lson.have()) {rt=rson; rt.l=tmp.l; rt.r=tmp.r; return;}
if (rson.have()) {rt=lson; rt.l=tmp.l; rt.r=tmp.r; return;}
rt.dis[][]=max( max( lson.dis[][]+rson.dis[][] , lson.dis[][]+rson.dis[][] ) , -INF);
rt.dis[][]=max( max( lson.dis[][]+rson.dis[][] , lson.dis[][]+rson.dis[][] ) , -INF);
rt.dis[][]=max( max( lson.dis[][]+rson.dis[][] , lson.dis[][]+rson.dis[][] ) , -INF);
rt.dis[][]=max( max( lson.dis[][]+rson.dis[][] , lson.dis[][]+rson.dis[][] ) , -INF);
rt.Dis[][]=max( lson.Dis[][] , max( lson.dis[][]+rson.Dis[][] , lson.dis[][]+rson.Dis[][] ) );
rt.Dis[][]=max( lson.Dis[][] , max( lson.dis[][]+rson.Dis[][] , lson.dis[][]+rson.Dis[][] ) );
rt.Dis[][]=max( rson.Dis[][] , max( rson.dis[][]+lson.Dis[][] , rson.dis[][]+lson.Dis[][] ) );
rt.Dis[][]=max( rson.Dis[][] , max( rson.dis[][]+lson.Dis[][] , rson.dis[][]+lson.Dis[][] ) );
}
inline void Update(int now) {update(tree[now],tree[ls],tree[rs]);}
inline void Modify(int now,int pos,char MP[])
{
int l=tree[now].l,r=tree[now].r;
if (l==r) {tree[now].modify(MP[]=='.',MP[]=='.'); return;}
int mid=(l+r)>>;
if (pos<=mid) Modify(ls,pos,MP); else Modify(rs,pos,MP);
Update(now);
}
inline SgtNode Query(int now,int L,int R)
{
int l=tree[now].l,r=tree[now].r;
if (l==L && R==r) return tree[now];
int mid=(l+r)>>; SgtNode ret;
if (R<=mid) return Query(ls,L,R);
else if (L>mid) return Query(rs,L,R);
else return update(ret,Query(ls,L,mid),Query(rs,mid+,R)),ret;
}
inline void Build(int now,int l,int r)
{
tree[now].l=l; tree[now].r=r;
if (l==r) return;
int mid=(l+r)>>;
Build(ls,l,mid); Build(rs,mid+,r);
}
}
using namespace SegmentTree;
namespace Divide
{
struct EdgeNode{int next,to;}edge[MAXN<<];
int head[MAXN],cnt=;
inline void AddEdge(int u,int v) {cnt++; edge[cnt].to=v; edge[cnt].next=head[u]; head[u]=cnt;}
inline void InsertEdge(int u,int v) {AddEdge(u,v); AddEdge(v,u);}
int deep[MAXN],fa[MAXN],size[MAXN],son[MAXN],pl[MAXN],dfn,top[MAXN];
inline void DFS_1(int now,int last)
{
size[now]=;
for (int i=head[now]; i; i=edge[i].next)
if (edge[i].to!=last)
{
fa[edge[i].to]=now;
deep[edge[i].to]=deep[now]+;
DFS_1(edge[i].to,now);
size[now]+=size[edge[i].to];
if (size[son[now]]<size[edge[i].to]) son[now]=edge[i].to;
}
}
inline void DFS_2(int now,int chain)
{
pl[now]=++dfn; top[now]=chain;
if (son[now]) DFS_2(son[now],chain);
for (int i=head[now]; i; i=edge[i].next)
if (edge[i].to!=fa[now] && edge[i].to!=son[now])
DFS_2(edge[i].to,edge[i].to);
}
inline void Modify(int pos,char MP[]) {SegmentTree::Modify(,pl[pos],MP);}
inline void GetAns(int x,int y)
{
SgtNode la,ra,ans; ra.l=ra.r=; la.l=la.r=;
while (top[x]!=top[y])
if (deep[top[x]]<deep[top[y]])
update(ra,Query(,pl[top[y]],pl[y]),ra),y=fa[top[y]];
else
update(la,Query(,pl[top[x]],pl[x]),la),x=fa[top[x]];
if (deep[x]<deep[y])
update(ra,Query(,pl[x],pl[y]),ra);
else
update(la,Query(,pl[y],pl[x]),la);
la.rever(); update(ans,la,ra);
printf("%d\n",max(ans.Dis[][],ans.Dis[][]));
}
inline void BuildTree() {for (int i=; i<=N; i++) Modify(i,mp[i]);}
}
using namespace Divide;
int main()
{
// freopen("fight.in","r",stdin); freopen("fight.out","w",stdout);
N=read(),M=read();
for (int x,y,i=; i<=N-; i++) x=read(),y=read(),InsertEdge(x,y);
Divide::DFS_1(,); Divide::DFS_2(,);
for (int i=; i<=N; i++) scanf("%s",mp[i]+);
SegmentTree::Build(,,N); Divide::BuildTree();
while (M--)
{
char opt[],MP[]; scanf("%s",opt); int L,R,p;
switch (opt[])
{
case 'Q' : L=read(),R=read(); Divide::GetAns(L,R); break;
case 'C' : p=read(); scanf("%s",MP+); Divide::Modify(p,MP); break;
}
}
return ;
}
【BZOJ-2325】道馆之战 树链剖分 + 线段树的更多相关文章
- BZOJ2325[ZJOI2011]道馆之战——树链剖分+线段树
题目描述 口袋妖怪(又名神奇宝贝或宠物小精灵)红/蓝/绿宝石中的水系道馆需要经过三个冰地才能到达馆主的面前,冰地中 的每一个冰块都只能经过一次.当一个冰地上的所有冰块都被经过之后,到下一个冰地的楼梯才 ...
- 【bzoj2325】[ZJOI2011]道馆之战 树链剖分+线段树区间合并
题目描述 给定一棵树,每个节点有上下两个格子,每个格子的状态为能走或不能走.m次操作,每次修改一个节点的状态,或询问:把一条路径上的所有格子拼起来形成一个宽度为2的长方形,从起点端两个格子的任意一个开 ...
- BZOJ.4034 [HAOI2015]树上操作 ( 点权树链剖分 线段树 )
BZOJ.4034 [HAOI2015]树上操作 ( 点权树链剖分 线段树 ) 题意分析 有一棵点数为 N 的树,以点 1 为根,且树点有边权.然后有 M 个 操作,分为三种: 操作 1 :把某个节点 ...
- BZOJ.1036 [ZJOI2008]树的统计Count ( 点权树链剖分 线段树维护和与最值)
BZOJ.1036 [ZJOI2008]树的统计Count (树链剖分 线段树维护和与最值) 题意分析 (题目图片来自于 这里) 第一道树链剖分的题目,谈一下自己的理解. 树链剖分能解决的问题是,题目 ...
- BZOJ 3672[NOI2014]购票(树链剖分+线段树维护凸包+斜率优化) + BZOJ 2402 陶陶的难题II (树链剖分+线段树维护凸包+分数规划+斜率优化)
前言 刚开始看着两道题感觉头皮发麻,后来看看题解,发现挺好理解,只是代码有点长. BZOJ 3672[NOI2014]购票 中文题面,题意略: BZOJ 3672[NOI2014]购票 设f(i)f( ...
- bzoj 4196 [Noi2015]软件包管理器 (树链剖分+线段树)
4196: [Noi2015]软件包管理器 Time Limit: 10 Sec Memory Limit: 512 MBSubmit: 2852 Solved: 1668[Submit][Sta ...
- bzoj 2157: 旅游【树链剖分+线段树】
裸的树链剖分+线段树 但是要注意一个地方--我WA了好几次才发现取完相反数之后max值和min值是要交换的-- #include<iostream> #include<cstdio& ...
- BZOJ 3589 动态树 (树链剖分+线段树)
前言 众所周知,90%90\%90%的题目与解法毫无关系. 题意 有一棵有根树,两种操作.一种是子树内每一个点的权值加上一个同一个数,另一种是查询多条路径的并的点权之和. 分析 很容易看出是树链剖分+ ...
- 【BZOJ2243】[SDOI2011]染色 树链剖分+线段树
[BZOJ2243][SDOI2011]染色 Description 给定一棵有n个节点的无根树和m个操作,操作有2类: 1.将节点a到节点b路径上所有点都染成颜色c: 2.询问节点a到节点b路径上的 ...
随机推荐
- iOS tableViewCell自适应高度 第三发类库
在github中有许多大牛封装好的第三发类库,其中有个自适应cell高度的类库 下载地址:https://github.com/gsdios/SDAutoLayout model类 commentsM ...
- 敏捷开发与jira
项目背景 项目是基于一套公司自主研发的平台做企业信息化的项目管理业务,经过两个里程碑的交付,已经在客户现场使用,每次版本都能按期交付,延迟较少,客户满意度也高. 项目开发过程采用的敏捷的方法,用类Sc ...
- 实战:考虑性能--Solr索引的schema设计
从 high level 的角度来看,schema.xml 结果如下,这个例子虽然不是一个真实的XML,但是简洁明了的传达了shema的概念. <schema> <types> ...
- 大数据系列(5)——Hadoop集群MYSQL的安装
前言 有一段时间没写文章了,最近事情挺多的,现在咱们回归正题,经过前面四篇文章的介绍,已经通过VMware安装了Hadoop的集群环境,相关的两款软件VSFTP和SecureCRT也已经正常安装了. ...
- C++STL - 类模板
类的成员变量,成员函数,成员类型,以及基类中如果包含参数化的类型,那么该类就是一个类模板 1.定义 template<typename 类型形参1, typename 类型形参2,...&g ...
- 原生js与jquery操作iframe
1 原生js获取iframe的window对象 //方法1 document.getElementById('iframeId').contentWindow; //方法2 window.frame ...
- 浏览器请求页面时Etag和cache的区别
etag:简单的说就是服务器收到客户端的请求后,根据返回的内容计算出来一个etag值,返回给客户端,当客户端下次再请求相同路径的时候会带上之前的etag值,服务器端会根据这次请求应该返回的内容计算出新 ...
- 用Lua扩展谷歌拼音输入法
谷歌拼音输入法最后一次更新是2013年,最近2年毫无动静,这个产品应该已经停了,不过这并不影响对它的使用,我一直喜欢它的简洁和稳定. 说不上来什么原因,忽然想起了摆弄摆弄谷歌拼音输入法的扩展特性(我经 ...
- java设计模式之原型模式
原型模式概念 该模式的思想就是将一个对象作为原型,对其进行复制.克隆,产生一个和原对象类似的新对象.java中复制通过clone()实现的.clone中涉及深.浅复制.深.浅复制的概念如下: ⑴浅复制 ...
- CF733D Kostya the Sculptor[贪心 排序]
D. Kostya the Sculptor time limit per test 3 seconds memory limit per test 256 megabytes input stand ...