hdu 4857 Little Devil I
http://acm.hdu.edu.cn/showproblem.php?pid=4897
题意:给你一棵树,边的颜色要么为白色,要么为黑色,初始每条边为白色,有三种操作
1、将u-v链上面的所有边的颜色翻转2、将u-v链上面所有邻接的边翻转(边上只有一个点在链上面)3、询问u->v上面有多少黑色的边树链剖分,线段树维护4个信息:
按dfs序建立线段树后,如果线段树内节点的区间为[l,r],则此节点维护树上dfs序[l,r]内的父边的信息
父边指 点与父节点之间的边
sum0:节点的父边属于重链且颜色为白色 的边数
sum1:节点的父边属于重链且颜色为黑色 的边数
rev1:节点的父边颜色是否被操作1取反 (实际只会用到属于轻链的边)
rev2:节点的子树中,与节点直接相连的属于轻链边 是否被操作2取反
操作1:直接取反,交换sum0和sum1,维护标记rev1
细节:最后u和v(dep[u]<dep[v])汇集到一条重链的时候,最后一次操作不包括u,因为点代表的是父边的信息
操作2:一条链的相邻边,除了对链上的点维护rev2操作外,
链最顶端的点如果是其父节点的重儿子,需要修改它的rev1
路径上每条重链最底端的点,如果它有重儿子,需要修改它重儿子的rev1
因为标记rev2只维护轻链
操作3:重链直接查,轻链呢?
在树链剖分往上跳的时候,跳轻链一定是只跳一条边
假设这条边连接了节点u和v,dep[u]<dep[v]
如果rev2[u]^rev2[v]^rev1[v] 为 true,则这条边为黑色
clj的题就是好哇!!!
#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm> using namespace std; #define N 100001 int n; int front[N],nxt[N<<],to[N<<],tot; int siz[N],dep[N],fa[N];
int bl[N],son[N];
int id[N],dy[N],cnt; bool big[N]; int sum0[N<<],sum1[N<<];
bool rev1[N<<],rev2[N<<]; int ans; void read(int &x)
{
x=; char c=getchar();
while(!isdigit(c)) c=getchar();
while(isdigit(c)) { x=x*+c-''; c=getchar(); }
} void add(int u,int v)
{
to[++tot]=v; nxt[tot]=front[u]; front[u]=tot;
to[++tot]=u; nxt[tot]=front[v]; front[v]=tot;
} void dfs1(int x)
{
siz[x]=;
for(int i=front[x];i;i=nxt[i])
if(to[i]!=fa[x])
{
fa[to[i]]=x;
dep[to[i]]=dep[x]+;
dfs1(to[i]);
siz[x]+=siz[to[i]];
}
} void dfs2(int x,int top)
{
bl[x]=top;
id[x]=++cnt;
dy[cnt]=x;
int y=;
for(int i=front[x];i;i=nxt[i])
if(to[i]!=fa[x] && siz[to[i]]>siz[y]) y=to[i];
if(y)
{
son[x]=y;
big[y]=true;
dfs2(y,top);
}
else return;
for(int i=front[x];i;i=nxt[i])
if(to[i]!=fa[x] && to[i]!=y) dfs2(to[i],to[i]);
} void down1(int k)
{
rev1[k<<]^=;
swap(sum0[k<<],sum1[k<<]);
rev1[k<<|]^=;
swap(sum0[k<<|],sum1[k<<|]);
rev1[k]^=;
} void down2(int k)
{
rev2[k<<]^=;
rev2[k<<|]^=;
rev2[k]^=;
} void Reverse(int k,int l,int r,int opl,int opr,int ty)
{
if(l>=opl && r<=opr)
{
if(ty==)
{
swap(sum1[k],sum0[k]);
rev1[k]^=;
}
else rev2[k]^=;
return;
}
int mid=l+r>>;
if(rev1[k]) down1(k);
if(rev2[k]) down2(k);
if(opl<=mid) Reverse(k<<,l,mid,opl,opr,ty);
if(opr>mid) Reverse(k<<|,mid+,r,opl,opr,ty);
if(ty==)
{
sum1[k]=sum1[k<<]+sum1[k<<|];
sum0[k]=sum0[k<<]+sum0[k<<|];
}
} int get_lca(int u,int v)
{
while(bl[u]!=bl[v])
{
if(dep[bl[u]]<dep[bl[v]]) swap(u,v);
u=fa[bl[u]];
}
return dep[u]<dep[v] ? u : v;
} bool point_query(int k,int l,int r,int x,int ty)
{
if(l==r) return ty== ? rev1[k] : rev2[k];
if(rev1[k]) down1(k);
if(rev2[k]) down2(k);
int mid=l+r>>;
if(x<=mid) return point_query(k<<,l,mid,x,ty);
return point_query(k<<|,mid+,r,x,ty);
} void query(int k,int l,int r,int opl,int opr)
{
if(l>=opl && r<=opr)
{
ans+=sum1[k];
return;
}
if(rev1[k]) down1(k);
if(rev2[k]) down2(k);
int mid=l+r>>;
if(opl<=mid) query(k<<,l,mid,opl,opr);
if(opr>mid) query(k<<|,mid+,r,opl,opr);
} void solve(int ty,int u,int v)
{
if(ty==)
{
while(bl[u]!=bl[v])
{
if(dep[bl[u]]<dep[bl[v]]) swap(u,v);
Reverse(,,n,id[bl[u]],id[u],);
u=fa[bl[u]];
}
if(dep[u]>dep[v]) swap(u,v);
if(u!=v) Reverse(,,n,id[u]+,id[v],);
}
else if(ty==)
{
int lca=get_lca(u,v);
if(lca!=u && lca!=v)
{
if(big[lca]) Reverse(,,n,id[lca],id[lca],);
}
else
{
if(dep[u]>dep[v]) swap(u,v);
if(big[u]) Reverse(,,n,id[u],id[u],);
}
while(bl[u]!=bl[v])
{
if(dep[bl[u]]<dep[bl[v]]) swap(u,v);
if(son[u]) Reverse(,,n,id[son[u]],id[son[u]],);
Reverse(,,n,id[bl[u]],id[u],);
u=fa[bl[u]];
}
if(dep[u]>dep[v]) swap(u,v);
if(son[v]) Reverse(,,n,id[son[v]],id[son[v]],);
Reverse(,,n,id[u],id[v],);
}
else
{
ans=;
while(bl[u]!=bl[v])
{
if(dep[bl[u]]<dep[bl[v]]) swap(u,v);
query(,,n,id[bl[u]],id[u]);
ans+=point_query(,,n,id[bl[u]],)^point_query(,,n,id[fa[bl[u]]],)^point_query(,,n,id[bl[u]],);
u=fa[bl[u]];
}
if(dep[u]>dep[v]) swap(u,v);
if(u!=v) query(,,n,id[u]+,id[v]);
printf("%d\n",ans);
}
} void build(int k,int l,int r)
{
sum0[k]=sum1[k]=;
rev1[k]=rev2[k]=false;
if(l==r)
{
sum0[k]=big[dy[l]];
return;
}
int mid=l+r>>;
build(k<<,l,mid);
build(k<<|,mid+,r);
sum0[k]=sum0[k<<]+sum0[k<<|];
} void clear()
{
tot=cnt=;
memset(front,,sizeof(front));
memset(son,,sizeof(son));
memset(big,false,sizeof(big));
} int main()
{
freopen("data.in","r",stdin);
freopen("my.out","w",stdout);
int T;
read(T);
int u,v;
int ty,m,lca;
while(T--)
{
clear();
read(n);
for(int i=;i<n;++i)
{
read(u); read(v);
add(u,v);
}
dfs1();
dfs2(,);
build(,,n);
read(m);
while(m--)
{
read(ty); read(u); read(v);
solve(ty,u,v);
}
}
return ;
}
Little Devil I
Time Limit: 16000/8000 MS (Java/Others) Memory Limit: 131072/131072 K (Java/Others)
Total Submission(s): 1087 Accepted Submission(s):
378
a devil. The devil always asks the king to do some crazy things. Although the
king used to be wise and beloved by his people. Now he is just like a boy in
love and can’t refuse any request from the devil. Also, this devil is looking
like a very cute Loli.
The devil likes to make thing in chaos. This
kingdom’s road system is like simply a tree(connected graph without cycle). A
road has a color of black or white. The devil often wants to make some change of
this system.
In details, we call a path on the tree from a to b consists
of vertices lie on the shortest simple path between a and b. And we say an edge
is on the path if both its two endpoints is in the path, and an edge is adjacent
to the path if exactly one endpoint of it is in the path.
Sometimes the
devil will ask you to reverse every edge’s color on a path or adjacent to a
path.
The king’s daughter, WJMZBMR, is also a cute loli, she is surprised
by her father’s lolicon-like behavior. As she is concerned about the
road-system’s status, sometimes she will ask you to tell there is how many black
edge on a path.
Initially, every edges is white.
number of the test cases.
For each test case, the first line contains an
integer n, which is the size of the tree. The vertices be indexed from 1.
On
the next n-1 lines, each line contains two integers a,b, denoting there is an
edge between a and b.
The next line contains an integer Q, denoting the
number of the operations.
On the next Q lines, each line contains three
integers t,a,b. t=1 means we reverse every edge’s color on path a to b. t=2
means we reverse every edge’s color adjacent to path a to b. t=3 means we query
about the number of black edge on path a to
b.
T<=5.
n,Q<=10^5.
Please use scanf,printf instead of
cin,cout,because of huge input.
line.
10
2 1
3 1
4 1
5 1
6 5
7 4
8 3
9 5
10 6
10
2 1 6
1 3 8
3 8 10
2 3 4
2 10 8
2 4 10
1 7 6
2 7 3
2 1 4
2 10 10
reverse color means change from white to black or vice virsa.
hdu 4857 Little Devil I的更多相关文章
- HDU 4857 拓扑排序 优先队列
n个数,已经有大小关系,现给m个约束,规定a在b之前,剩下的数要尽可能往前移.输出序列 大小关系显然使用拓扑结构,关键在于n个数本身就有大小关系,那么考虑反向建图,优先选择值最大的入度为零的点,这样得 ...
- HDU 4857
HDU 4857 (反向拓扑排序 + 优先队列) 糟糕的事情发生啦,现在大家都忙着逃命.但是逃命的通道很窄,大家只能排成一行. 现在有n个人,从1标号到n.同时有一些奇怪的约束条件,每个都形如:a必须 ...
- HDU 4857 topological_sort
逃生 Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) Total Submission ...
- codeforce E - Minimal Labels+hdu 4857
两个题目的意思差不多 都是希望得出的拓扑序如果有多种 要求输出字典序小的情况 这里引用一个大佬的博客 关于为什么不能直接建图然后用小根堆解决这个问题(http://blog.csdn.net/rgno ...
- HDU 4857 逃生 (反向拓扑排序 & 容器实现)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4857 逃生 Time Limit: 2000/1000 MS (Java/Others) Mem ...
- HDU 4857 逃生 (优先队列+反向拓扑)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4857 解题报告:有n个点,有m个条件限制,限制是像这样的,输入a b,表示a必须排在b的前面,如果不 ...
- HDU 4897 Little Devil I(树链剖分)(2014 Multi-University Training Contest 4)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4897 Problem Description There is an old country and ...
- hdu 4857 逃生
题目连接 http://acm.hdu.edu.cn/showproblem.php?pid=4857 逃生 Description 糟糕的事情发生啦,现在大家都忙着逃命.但是逃命的通道很窄,大家只能 ...
- (hdu) 4857 逃生 (拓扑排序+优先队列)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4857 Problem Description 糟糕的事情发生啦,现在大家都忙着逃命.但是逃命的通道很窄 ...
随机推荐
- 巧用cheerio重构grunt-inline
grunt-inline是楼主之前写的一个插件,主要作用是把页面带了__inline标记的资源内嵌到html页面去.比如下面的这个script标签. <script src="main ...
- C#_委托
委托属于C#中的新名词,它的应用也非常广泛,例如事件就是委托最简单而又直接的例子. 那么首先说说什么是委托,其实委托在用过C或者C++的人看来就是函数指针,不过使用C#的大多数人都没有用过这两门语言, ...
- 【原】python3.7 无法pip安装提示ssl错误解决方案
问题 pip is configured with locations that require TLS/SSL, however the ssl module in Python is not av ...
- 甲题题解-1116. Come on! Let’s C (20)-(素数筛选法)
用vis标记出现过的id,checked标记询问过的id.至于如何判断排名为素数,用素数筛选法预处理一下即可,水题. #include <iostream> #include <cs ...
- Linux内核分析第五章读书笔记
第五章 系统调用 在操作系统中,内核提供了用户进程与内核进行交互的一组接口,这些接口在应用程序和内核之间扮演了使者的角色,保证系统稳定可靠,避免应用程序肆意妄行. 5.1 与内核通信 系统调用在用户空 ...
- 读书笔记(chapter18)
调试 18.1准备开始 18.2内核中的bug 1.从隐藏在源代码中的错误到展现在目击者面前的bug,往往是经历一系列连锁反应的事件才可能触发的 18.3通过打印来调试 1.健壮性 健壮性是print ...
- vsphere web client 使用中文的解决办法
1. 很多网站这么说的: vsphere web client的默认URL为:https://hostname:9443/vsphere-client 可以在URL后面加上一个参数来指定区域语言 英语 ...
- Python day7之mysql
写在前面: 由于毕业论文撰写和答辩耽搁了几个月,但是在这几个月没有放弃学习Python,就是没有时间写博客.进行我们主要对数据库mysql的操作指令集的学习. 一.mysql术语 Mysql是最流行的 ...
- hdwiki 参考资料改成可点击跳转的(默认不能点)
(1)首先在view这个文件夹找到viewdoc.htm文件.(2)在viewdoc.htm源码中搜索“<dd> <span>[{eval echo ($i+1)}]”找到下面 ...
- 洛谷P1450 [HAOI2008]硬币购物(背包问题,容斥原理)
洛谷题目传送门 我实在是太弱了,第一次正儿八经写背包DP,第一次领会如此巧妙的容斥原理的应用...... 对每次询问都做一遍多重背包,显然T飞,就不考虑了 关键就在于每次询问如何利用重复的信息 我这么 ...