bzoj3637 CodeChef SPOJ - QTREE6 Query on a tree VI 题解
题意:
一棵n个节点的树,节点有黑白两种颜色,初始均为白色。两种操作:1.更改一个节点的颜色;2.询问一个节点所处的颜色相同的联通块的大小。
思路:
1.每个节点记录仅考虑其子树时,假设其为黑色时所处的黑色联通块的大小和假设其为白色时所处的白色联通块的大小(树状数组维护)。
2.查询时找到深度最小的、与该点颜色相同的且两点之间的点颜色均与这两点相同(两点可以重合)(不妨称它为最远祖先)的答案。
3.修改时应该修改该节点的父亲至最远祖先的父亲上的值。
4.用树链剖分和树状数组维护。
5.寻找最远祖先时,跳重链,树状数组维护当前颜色(0为白 1为黑),查询重链上的颜色和,判断是否整段相同。是则继续向上跳,否则二分祖先。
反思:
1.越上面的dfn越小,二分开始打反了(二分看了Po姐的)
2.树链剖分有些生疏,开始时打错死循了。树状数组的应用不熟练。
3.开始时修改时是边找最远祖先边修改的,结果在根周围时出现了问题。
4.根在修改、查询时都特判一下,重链的端点为最远祖先时也特判一下。
代码:
#include<cstdio>
#define M 100005
#define swap(x,y) u=x,x=y,y=u
int n,u,cnt,dfn,p[M],v[M<<],id[M],co[M],sz[M],di[M],dep[M],top[M],hea[M],nex[M<<],ans[M][];
bool col[M]; void ins(int x,int y) { v[++cnt]=y,nex[cnt]=hea[x],hea[x]=cnt; }
void Add(int x,int y) { for (;x<=n;x+=x&-x) co[x]+=y; }
int Ask(int x) { int s=; for (;x;x-=x&-x) s=s+co[x]; return s; }
void add(int x,int y,bool c) { for (;x<=n;x+=x&-x) ans[x][c]+=y; }
int ask(int x,bool c) { int s=; for (;x;x-=x&-x) s=s+ans[x][c]; return s;}
bool pd(int x,int y,bool c)
{ return c && (Ask(y)-Ask(x-))==(y-x+) || (!c) && (Ask(x-)==Ask(y)); }
int read()
{
int x=; char ch=getchar();
for (;ch< || ch>;ch=getchar());
for (;ch> && ch<;ch=getchar()) x=(x<<)+(x<<)+ch-;
return x;
} void dfs(int x)
{
sz[x]=;
for (int i=hea[x],y;i;i=nex[i])
if ((y=v[i])^p[x]) p[y]=x,dep[y]=dep[x]+,dfs(y),sz[x]+=sz[y];
} void dfs(int x,int chain)
{
int i,k=,y;
top[x]=chain,id[x]=++dfn,di[dfn]=x;
for (i=hea[x];i;i=nex[i])
if ((y=v[i])^p[x] && sz[y]>sz[k]) k=y;
if (!k) return; dfs(k,chain);
for (i=hea[x];i;i=nex[i])
if ((y=v[i])^p[x] && y^k) dfs(y,y);
} void change(int x,int y,int v,bool c)
{
for (;top[x]!=top[y];x=p[top[x]])
{
if (dep[top[x]]<dep[top[y]]) swap(x,y);
add(id[top[x]],v,c),add(id[x]+,-v,c);
}
if (dep[x]<dep[y]) swap(x,y);
add(id[y],v,c),add(id[x]+,-v,c);
} int find(int x,bool c)
{
int l,r,t,y,mid;
for (;x^;x=p[y])
{
l=id[y=top[x]],t=r=id[x];
if (!pd(l,r,c))
{
while (l+<r)
if (pd((mid=(l+r)>>),t,c)) r=mid; else l=mid+;
if (pd(l,t,c)) return l; return r;
}
if (col[p[y]]^c) return l;
}
return ;
} int main()
{
n=read(); int i,x,y,m;
for (i=;i<n;++i) x=read(),y=read(),ins(x,y),ins(y,x);
dfs(p[]=),dfs(,),add(,,);
for (i=;i<=n;++i) add(id[i],sz[i],),add(id[i]+,-sz[i],);
for (m=read();m--;)
{
i=read(),x=read();
if (i)
{
i=col[x];if (x-) change(p[x],p[di[find(x,i)]],-ask(id[x],i),i);
if (i) Add(id[x],-); else Add(id[x],); col[x]^=;
i=col[x];if (x-) change(p[x],p[di[find(x,i)]],ask(id[x],i),i);
}
else i=col[x],printf("%d\n",ask(find(x,i),i));
}
return ;
}
bzoj3637 CodeChef SPOJ - QTREE6 Query on a tree VI 题解的更多相关文章
- SPOJ QTREE6 Query on a tree VI 树链剖分
题意: 给出一棵含有\(n(1 \leq n \leq 10^5)\)个节点的树,每个顶点只有两种颜色:黑色和白色. 一开始所有的点都是黑色,下面有两种共\(m(1 \leq n \leq 10^5) ...
- QTREE6 - Query on a tree VI 解题报告
QTREE6 - Query on a tree VI 题目描述 给你一棵\(n\)个点的树,编号\(1\)~\(n\).每个点可以是黑色,可以是白色.初始时所有点都是黑色.下面有两种操作请你操作给我 ...
- SPOJ 16549 - QTREE6 - Query on a tree VI 「一种维护树上颜色连通块的操作」
题意 有操作 $0$ $u$:询问有多少个节点 $v$ 满足路径 $u$ 到 $v$ 上所有节点(包括)都拥有相同的颜色$1$ $u$:翻转 $u$ 的颜色 题解 直接用一个 $LCT$ 去暴力删边连 ...
- SPOJ QTREE Query on a tree VI
You are given a tree (an acyclic undirected connected graph) with n nodes. The tree nodes are number ...
- SP16549 QTREE6 - Query on a tree VI LCT维护颜色联通块
\(\color{#0066ff}{ 题目描述 }\) 给你一棵n个点的树,编号1~n.每个点可以是黑色,可以是白色.初始时所有点都是黑色.下面有两种操作请你操作给我们看: 0 u:询问有多少个节点v ...
- [QTree6]Query on a tree VI
Description: 给你一棵n个点的树,编号1~n.每个点可以是黑色,可以是白色.初始时所有点都是黑色.下面有两种操作请你操作给我们看: 0 u:询问有多少个节点v满足路径u到v上所有节点(包括 ...
- 洛谷SP16549 QTREE6 - Query on a tree VI(LCT)
洛谷题目传送门 思路分析 题意就是要维护同色连通块大小.要用LCT维护子树大小就不说了,可以看看蒟蒻的LCT总结. 至于连通块如何维护,首先肯定可以想到一个很naive的做法:直接维护同色连通块,每次 ...
- SP16549 QTREE6 - Query on a tree VI(LCT)
题意翻译 题目描述 给你一棵n个点的树,编号1~n.每个点可以是黑色,可以是白色.初始时所有点都是黑色.下面有两种操作请你操作给我们看: 0 u:询问有多少个节点v满足路径u到v上所有节点(包括)都拥 ...
- bzoj 3637: Query on a tree VI 树链剖分 && AC600
3637: Query on a tree VI Time Limit: 8 Sec Memory Limit: 1024 MBSubmit: 206 Solved: 38[Submit][Sta ...
随机推荐
- import和from .xxx import *的一点重要区别
import zzz 不会导入zzz中已导入的模块 from .xxx import * 会导入xxx中已导入的模块 特别注意: 使用logging时,x模块导入了log.py,y模块导入了log.p ...
- qconbeijing2016
http://2016.qconbeijing.com/schedule 大会日程 2016年04月21日 星期四 09:15 开场致辞 地点 1号厅 主题演讲 工程效率提升 业务核心架构 容器集 ...
- 证明碰撞集问题(Hitting Set)是NP-complete
证明碰撞集问题(Hitting Set)是NP-complete Problem In the HITTING SET problem, we are given a family of sets { ...
- hdu4003/蓝桥杯 金属采集
思路: 树形dp + 分组背包dp. 参考https://www.cnblogs.com/kuangbin/archive/2012/08/29/2661928.html 实现: #include & ...
- poj2377 Bad Cowtractors
思路: 最大生成树. 实现: #include <iostream> #include <cstdio> #include <vector> #include &l ...
- js数组常用方法整理
学疏才浅,若有不对的地方,希望大家可以帮忙指正出来. 1. Array.push(),向数组的末尾添加一个或多个元素,并返回新的数组长度.原数组改变. 2. Array.pop(),删除并返回数组的最 ...
- upload 上传 加token 在 :headers='headers' 注意 不要直接写$refs.upload.headers = {} 这样vue会警告 修改组件内部变量
upload 上传 加token 在 :headers='headers' 注意 不要直接写$refs.upload.headers = {} 这样vue会警告 修改组件内部变量 <Upload ...
- vue 数据没有驱动视图?
Part.1 问题 数据改变,视图却没有根据数据而改变. 原因在于,数据并不在 vue 监听范围之内,vue 只对事先在 data 中声明的变量丶对象等类型数据进行监听 Part.2 例子 < ...
- Windows下使用ffmpeg与java实现截取视频缩略图
[ffmpeg.exe可执行文件获取]: 网上搜索后得到编译好的ffmpeg文件,下载地址:http://download.csdn.net/source/453719 [安装]: 将下载的文件解压, ...
- HTML 5 <a> 标签
href 属性 定义和用法 href 属性规定链接的目标地址. 如果未使用 href 属性,则 <a> 标签不是链接,而是链接的占位符. HTML 4.01 与 HTML 5 之间的差异 ...