大致题意: 给一棵树,每个节点最开始都是黑色,有两种操作,1.询问树中相距最远的一对黑点的距离 2.反转一个节点的颜色

一种做法:

  建立出树的括号序列,类似这样: [A[B][C]],所以长度为3*n

  假如我们要询问AC间的距离,提取出中间的括号:[]],匹配消去后得到],其长度就是距离.

  现在我们要做的就是修改点的状态,并且动态维护答案.要用到一些求与绝对值相关的式子的技巧.

 /**************************************************************
Problem: 1095
User: idy002
Language: C++
Result: Accepted
Time:2176 ms
Memory:55548 kb
****************************************************************/ #include <cstdio>
#define min(a,b) ((a)<(b)?(a):(b))
#define max(a,b) ((a)>(b)?(a):(b))
#define N 100010
#define M N<<1
#define oo 0x3f3f3f3f
#define fprintf(...) struct Node {
int v[], c, e, a;
int lf, rg;
Node *ls, *rs;
void init( int type ) {
if( type== ) {
a = ;
e = false;
c = ;
for( int t=; t<; t++ ) v[t]=;
} else if( type==- ) { // (1,0)
a = -;
e = true;
c = ;
v[] = v[] = v[] = v[] = v[] = ;
v[] = v[] = -;
} else { // (0,1)
a = -;
e = true;
c = ;
v[] = v[] = v[] = v[] = v[] = ;
v[] = v[] = -;
}
}
void update() {
e = ls->e && rs->e;
v[] = ls->v[] + rs->v[];
v[] = ls->v[] + rs->v[];
v[] = max( ls->v[]+rs->v[], ls->v[]+rs->v[] );
if( !ls->e && !rs->e ) {
v[] = max( ls->v[], ls->v[]+rs->v[] );
v[] = max( ls->v[], max( ls->v[]+rs->v[], ls->v[]+rs->v[] ) );
v[] = max( rs->v[], ls->v[]+rs->v[] );
v[] = max( rs->v[], max( ls->v[]+rs->v[], ls->v[]+rs->v[] ) );
a = max( max(ls->v[]+rs->v[],ls->v[]+rs->v[]), max(ls->a,rs->a) );
} else if( !ls->e ) {
v[] = ls->v[];
v[] = ls->v[];
v[] = ls->v[]+rs->v[];
v[] = max( ls->v[]+rs->v[], ls->v[]+rs->v[] );
a = ls->a;
} else if( !rs->e ) {
v[] = ls->v[]+rs->v[];
v[] = max( ls->v[]+rs->v[], ls->v[]+rs->v[] );
v[] = rs->v[];
v[] = rs->v[];
a = rs->a;
} else {
a = -;
}
}
void reverse( int pos ) {
if( lf==rg ) {
if( c ) {
c = ;
a = -;
e = true;
} else {
c = ;
a = ;
e = false;
}
return;
}
int mid=(lf+rg)>>;
if( pos<=mid ) ls->reverse(pos);
else rs->reverse(pos);
update();
}
}pool[N**], *tail=pool, *root; int n, m;
int head[N], dest[M], next[M], etot;
int dfn[N], sgn[N*], fat[N], idc; void adde( int u, int v ) {
etot++;
next[etot] = head[u];
dest[etot] = v;
head[u] = etot;
}
void dfs( int u ) {
sgn[++idc] = ;
dfn[u] = ++idc;
fprintf( stderr, "[" );
fprintf( stderr, "%d", u );
for( int t=head[u]; t; t=next[t] ) {
int v=dest[t];
if( v==fat[u] ) continue;
fat[v] = u;
dfs(v);
}
sgn[++idc] = -;
fprintf( stderr, "]" );
}
Node *build( int lf, int rg ) {
Node *nd = ++tail;
nd->lf=lf, nd->rg=rg;
if( lf==rg ) {
nd->init( sgn[lf] );
} else {
int mid=(lf+rg)>>;
nd->ls = build( lf, mid );
nd->rs = build( mid+, rg );
nd->update();
}
fprintf( stderr, "[%d,%d] a=%2d e=%d v[0~6] = %2d %2d %2d %2d %2d %2d %2d\n",
lf, rg, nd->a, nd->e, nd->v[], nd->v[],
nd->v[], nd->v[], nd->v[], nd->v[], nd->v[] );
return nd;
}
int main() {
scanf( "%d", &n );
for( int i=,u,v; i<n; i++ ) {
scanf( "%d%d", &u, &v );
adde( u, v );
adde( v, u );
}
for( int i=; i<=n+n+n; i++ )
fprintf( stderr, "%d", i% );
fprintf( stderr, "\n" );
fat[] = ;
dfs();
fprintf( stderr, "\n" );
root = build( , idc );
scanf( "%d", &m );
for( int i=,u; i<=m; i++ ) {
char ch[];
scanf( "%s\n", ch );
if( ch[]=='G' ) {
printf( "%d\n", root->a );
} else {
scanf( "%d", &u );
root->reverse(dfn[u]);
}
}
}

另一种做法大概是用点分,然后用堆维护最值.

bzoj 1095 括号序列求两点距离的更多相关文章

  1. [BZOJ 4350]括号序列再战猪猪侠 题解(区间DP)

    [BZOJ 4350]括号序列再战猪猪侠 Description 括号序列与猪猪侠又大战了起来. 众所周知,括号序列是一个只有(和)组成的序列,我们称一个括号 序列S合法,当且仅当: 1.( )是一个 ...

  2. bzoj 4244 括号序列dp

    将各种情况绕环等看作括号序列,括号内的区域上下都需要累加答案,左右也是 f[i][j] 代表 前i个车站已经处理完的有j个左括号的最小权值 我们可以发现,更新的来源来自于 i-1, 和 i 将上 描述 ...

  3. bzoj 2209 括号序列

    反转操作 + 翻转操作 = 对称操作 因为上面三个操作都是自己的逆操作,所以我们只需要实现对称操作和反转操作,就可以搞定翻转操作. #include <cstdio> #include & ...

  4. BZOJ 1095 捉迷藏(线段树维护括号序列)

    对于树的一个括号序列,树上两点的距离就是在括号序列中两点之间的括号匹配完之后的括号数... 由此可以得出线段树的做法.. #include<cstdio> #include<iost ...

  5. hdu 6299 Balanced Sequence (括号序列,贪心)

    大意: 记$f(t)$表示字符串$t$的最长括号匹配子序列, 给定n个括号序列, 求它们重排后的最大f(t). 首先可以注意到一个括号序列中已经匹配的可以直接消去, 一定不会影响最优解. 那么这样最终 ...

  6. NX二次开发-UFUN计算两点距离UF_VEC3_distance

    NX11+VS2013 #include <uf.h> #include <uf_curve.h> #include <uf_vec.h> UF_initializ ...

  7. 【BZOJ】1095: [ZJOI2007]Hide 捉迷藏 括号序列+线段树

    [题目]BZOJ 1095 [题意]给定n个黑白点的树,初始全为黑点,Q次操作翻转一个点的颜色,或询问最远的两个黑点的距离,\(n \leq 10^5,Q \leq 5*10^5\). [算法]括号序 ...

  8. bzoj 1095 [ZJOI2007]Hide 捉迷藏(括号序列+线段树)

    [题目链接] http://www.lydsy.com/JudgeOnline/problem.php?id=1095 [题意] 给定一棵树,树上颜色或白或黑而且可以更改,多个询问求最远黑点之间的距离 ...

  9. BZOJ.1095.[ZJOI2007]捉迷藏(线段树 括号序列)

    BZOJ 洛谷 对树DFS得到括号序列.比如这样一个括号序列:[A[B[E][F[H][I]]][C][D[G]]]. 那比如\(D,E\)间的最短距离,就是将\(D,E\)间的括号序列取出:][[] ...

随机推荐

  1. kafka系列一、kafka安装及部署、集群搭建

    一.环境准备 操作系统:Cent OS 7 Kafka版本:kafka_2.10 Kafka官网下载:请点击 JDK版本:1.8.0_171 zookeeper-3.4.10 二.kafka安装配置 ...

  2. mysql重置登录密码

    1.停止mysql服务. services.msc进入服务界面 停止mysql服务 2.打开一个cmd窗口. 输入mysqld --skip-grant-tables 启动了一个新的mysql服务 跳 ...

  3. nagios系列(六)之nagios实现对服务器cpu温度的监控

    1.安装硬件传感器监控软件sensors yum install -y lm_sensors* 2.运行sensors-detect进行传感器检测 ##一路回车即可 Do you want to ov ...

  4. sqlserver2008r2数据库关联孤立账号的方法

    重新安装mssql数据库后可能有这样的问题,     在数据还原到其他服务器时,登录名会丢失,重建非常麻烦     特別是数据库用户较多的的情况,其实这个问题解决非常简单,     sql serve ...

  5. appium运行报错java.net.SocketException: socket write error

    这个错我调了 快两天一点头绪没有,脚本正常跑没问题,但是就是控制台输出信息报错,没法定位问题在哪.报错如图: 虽然这个报错不影响测试结果,但是本人有强迫症,一定要查出究竟: 我的尝试: 1.那天试验, ...

  6. python 全栈开发,Day46(列表标签,表格标签,表单标签,css的引入方式,css选择器)

    一.列表标签 列表标签分为三种. 1.无序列表<ul>,无序列表中的每一项是<li> 英文单词解释如下: ul:unordered list,“无序列表”的意思. li:lis ...

  7. 【C++ Primer 第六章】 1. 定义模板

    类模板 题目描述:实现StrBlob的模板版本. /* Blob.h */ #include<iostream> #include<vector> #include<in ...

  8. Codeforces 891C Envy

    Envy 感觉这种最小生成树上的啥题都差不多的解法.. #include<bits/stdc++.h> #define LL long long #define fi first #def ...

  9. BZOJ1078 [SCOI2008]斜堆 堆

    欢迎访问~原文出处——博客园-zhouzhendong 去博客园看该题解 题目传送门 - BZOJ1078 题意概括 斜堆(skew heap)是一种常用的数据结构.它也是二叉树,且满足与二叉堆相同的 ...

  10. 【Java】 剑指offer(20) 表示数值的字符串

    本文参考自<剑指offer>一书,代码采用Java语言. 更多:<剑指Offer>Java实现合集   题目 请实现一个函数用来判断字符串是否表示数值(包括整数和小数).例如, ...