原文链接http://www.cnblogs.com/zhouzhendong/p/8081514.html


题目传送门 - BZOJ3862


题意概括

  一棵树,n个点,边权为黑或者白,支持3重操作:

  1.链上颜色翻转

  2.对于一条链,把有一个点在这条链上的边全部翻转颜色

  3.询问一条链上有多少黑色。


题解

  毒瘤题。

  对于1、3都是基础操作,很简单。

  主要是2.

  2的话,只需要打区间打标记,表示那些点的连向轻儿子的边全部翻转。然后修改的时候还有一堆特判(具体看代码)

  这题数据坑。

  有a==b的情况,小心了。我被坑了一个下午。

  我是怎么找到这个数据的呢?

  if (a==b) OLE();

void OLE(){
int a[100];
for (int i=23333;;i=i*i)
printf("233");
}

  


代码

#include <cstring>
#include <cstdio>
#include <cstdlib>
#include <algorithm>
#include <cmath>
using namespace std;
bool isd(char ch){return '0'<=ch&&ch<='9';}
int read(){
int x=0;
char ch=getchar();
while (!isd(ch))
ch=getchar();
while (isd(ch))
x=(x<<1)+(x<<3)+ch-48,ch=getchar();
return x;
}
const int N=100005;
struct Gragh{
int cnt,y[N*2],nxt[N*2],fst[N];
void clear(){
cnt=0;
memset(fst,0,sizeof fst);
}
void add(int a,int b){
y[++cnt]=b,nxt[cnt]=fst[a],fst[a]=cnt;
}
}g;
int T,n,q,cnp;
int size[N],fa[N],depth[N],son[N],p[N],ap[N],top[N];
void Get_Gen_Info(int rt,int pre,int d){
size[rt]=1,son[rt]=-1,fa[rt]=pre,depth[rt]=d;
for (int i=g.fst[rt];i;i=g.nxt[i])
if (g.y[i]!=pre){
int s=g.y[i];
Get_Gen_Info(s,rt,d+1);
size[rt]+=size[s];
if (son[rt]==-1||size[s]>size[son[rt]])
son[rt]=s;
}
}
void Get_Top(int rt,int tp){
top[rt]=tp;
ap[p[rt]=++cnp]=rt;
if (!~son[rt])
return;
Get_Top(son[rt],tp);
for (int i=g.fst[rt];i;i=g.nxt[i]){
int s=g.y[i];
if (s!=fa[rt]&&s!=son[rt])
Get_Top(s,s);
}
}
const int S=N*4;
struct Tree{
int sum,size,r,sr;
}t[S];
void build(int rt,int L,int R){
t[rt].size=R-L+1,t[rt].sum=t[rt].r=t[rt].sr=0;
if (L==R)
return;
int mid=(L+R)>>1,ls=rt<<1,rs=ls|1;
build(ls,L,mid);
build(rs,mid+1,R);
}
void pushup(int rt){
int ls=rt<<1,rs=ls|1;
t[rt].sum=t[ls].sum+t[rs].sum;
}
void pushdown(int rt){
int ls=rt<<1,rs=ls|1;
int &sr=t[rt].sr,&r=t[rt].r;
if (sr)
t[ls].sr^=1,t[rs].sr^=1,sr=0;
if (r){
t[ls].r^=1,t[ls].sum=t[ls].size-t[ls].sum;
t[rs].r^=1,t[rs].sum=t[rs].size-t[rs].sum;
r=0;
}
}
void update(int rt,int L,int R,int xL,int xR,int op){
if (R<xL||L>xR)
return;
if (xL<=L&&R<=xR){
if (op==1){
t[rt].sum=t[rt].size-t[rt].sum;
t[rt].r^=1;
}
else
t[rt].sr^=1;
return;
}
pushdown(rt);
int mid=(L+R)>>1,ls=rt<<1,rs=ls|1;
update(ls,L,mid,xL,xR,op);
update(rs,mid+1,R,xL,xR,op);
pushup(rt);
}
int query(int rt,int L,int R,int xL,int xR){
if (R<xL||L>xR)
return 0;
if (xL<=L&&R<=xR)
return t[rt].sum;
pushdown(rt);
int mid=(L+R)>>1,ls=rt<<1,rs=ls|1;
return query(ls,L,mid,xL,xR)+query(rs,mid+1,R,xL,xR);
}
int asksr(int rt,int L,int R,int pos){
if (L==R)
return t[rt].sr;
pushdown(rt);
int mid=(L+R)>>1,ls=rt<<1,rs=ls|1;
if (pos<=mid)
return asksr(ls,L,mid,pos);
else
return asksr(rs,mid+1,R,pos);
}
void Tupdate(int a,int b,int op){
if (a==b)
return;
int f1=top[a],f2=top[b];
while (f1!=f2){
if (depth[f1]<depth[f2])
swap(f1,f2),swap(a,b);
update(1,1,n,p[f1],p[a],op);
if (op==2){
update(1,1,n,p[f1],p[f1],1);
if (~son[a])
update(1,1,n,p[son[a]],p[son[a]],1);
}
a=fa[f1],f1=top[a];
}
if (depth[a]>depth[b])
swap(a,b);
if (op==2){
update(1,1,n,p[a],p[b],2);
update(1,1,n,p[a],p[a],1);
if (~son[b])
update(1,1,n,p[son[b]],p[son[b]],1);
}
else if (a!=b)
update(1,1,n,p[son[a]],p[b],op);
}
int Tquery(int a,int b){
int f1=top[a],f2=top[b],ans=0;
while (f1!=f2){
if (depth[f1]<depth[f2])
swap(f1,f2),swap(a,b);
if (f1!=a)
ans+=query(1,1,n,p[son[f1]],p[a]);
ans+=query(1,1,n,p[f1],p[f1])^asksr(1,1,n,p[fa[f1]]);
a=fa[f1],f1=top[a];
}
if (depth[a]>depth[b])
swap(a,b);
if (a!=b)
ans+=query(1,1,n,p[son[a]],p[b]);
return ans;
}
int main(){
T=read();
while (T--){
n=read();
g.clear();
for (int i=1,a,b;i<n;i++){
a=read(),b=read();
g.add(a,b),g.add(b,a);
}
cnp=0;
Get_Gen_Info(1,0,0);
Get_Top(1,1);
build(1,1,n);
q=read();
while (q--){
int op,a,b;
op=read(),a=read(),b=read();
if (a==b){
if (op==3)
puts("0");
if (op==2){
update(1,1,n,p[a],p[a],2);
if (~son[a])
update(1,1,n,p[son[a]],p[son[a]],1);
if (fa[a])
update(1,1,n,p[a],p[a],1);
}
continue;
}
if (op<3)
Tupdate(a,b,op);
if (op==3)
printf("%d\n",Tquery(a,b));
}
}
return 0;
}

  

BZOJ3862 Little Devil I 树链剖分的更多相关文章

  1. hdu_4897_Little Devil I(树链剖分)

    题目连接:http://acm.hdu.edu.cn/showproblem.php?pid=4897 题意:有三种操作,1是在树上的两个节点之间的路径改变当前的颜色,2是改变树上有且只有一个端点在u ...

  2. BZOJ3862Little Devil I——树链剖分+线段树

    题目大意: 给一棵树,每条边可能是黑色或白色(起始都是白色),有三种操作: 1.将u到v路径上所有边颜色翻转(黑->白,白->黑) 2.将只有一个点在u到v路径上的边颜色翻转 3.查询u到 ...

  3. hdu 4897 Little Devil I (树链剖分+线段树)

    题目链接:  http://acm.hdu.edu.cn/showproblem.php?pid=4897 题意: 给你一棵树,一开始每条边都是白色,有三种操作: 1.将 u - v路径上的边转换颜色 ...

  4. HDU 4897 Little Devil I 树链剖分+线段树

    Little Devil I Problem Description There is an old country and the king fell in love with a devil. T ...

  5. 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 ...

  6. 树链剖分+线段树 HDOJ 4897 Little Devil I(小恶魔)

    题目链接 题意: 给定一棵树,每条边有黑白两种颜色,初始都是白色,现在有三种操作: 1 u v:u到v路径(最短)上的边都取成相反的颜色 2 u v:u到v路径上相邻的边都取成相反的颜色(相邻即仅有一 ...

  7. HDU4897 (树链剖分+线段树)

    Problem Little Devil I (HDU4897) 题目大意 给定一棵树,每条边的颜色为黑或白,起始时均为白. 支持3种操作: 操作1:将a->b的路径中的所有边的颜色翻转. 操作 ...

  8. hdu 4897 树链剖分(重轻链)

    Little Devil I Time Limit: 16000/8000 MS (Java/Others)    Memory Limit: 131072/131072 K (Java/Others ...

  9. BZOJ 3626: [LNOI2014]LCA [树链剖分 离线|主席树]

    3626: [LNOI2014]LCA Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 2050  Solved: 817[Submit][Status ...

随机推荐

  1. Vuex与axios介绍

    Vuex集中式状态管理里架构 axios (Ajax) Vuex集中式状态管理架构 -简单介绍: vuex是一个专门为Vue.js设计的集中式状态管理架构. 我们把它理解为在data中需要共享给其他组 ...

  2. 解决layui选中项下一页清空问题

    项目中遇到给用户在所有产品中匹配一部分产品.用layui 第一页选好之后到第二页再选,等回到第一页时之前选择的都没了,解决这个问题的办法如下: //勾选的产品id集合 var chooseAdids ...

  3. Golang的单引号、双引号与反引号

    Go语言的字符串类型string在本质上就与其他语言的字符串类型不同: Java的String.C++的std::string以及Python3的str类型都只是定宽字符序列 Go语言的字符串是一个用 ...

  4. dnsmasq详解&手册

    Dnsmasq为小型网络提供网络基础设施:DNS,DHCP,路由器通告和网络引导.它被设计为轻量级且占用空间小,适用于资源受限的路由器和防火墙.它还被广泛用于智能手机和便携式热点的共享,并支持虚拟化框 ...

  5. VBS计时器2

    打开计时器,如果点击暂停,会显示你刚才事物所用的时间(以分钟为单位) dim c //控制循环 c= vbyes while c<>vbno dim a a= 60*hour(now)+m ...

  6. 创建表空间、新增用户、给用户赋予DBA权限 、删除用户下的上有数据表

    正文原创 一:查询数据库实例有多少用户: [oracle@localhost ~]$ sqlplus / as sysdba; SQL*Plus: Release 11.2.0.3.0 Product ...

  7. Confluence 6 字符集编码的问题解决

    如果你的 Confluence 站点的字符集没有被正确配置,你可能会遇到下面的问题: Non-ASCII 字符将会显示为问号(?) Non-ASCII 字符集的页面链接将不能工作 单一字符将会被显示为 ...

  8. Confluence 6 附件存储提取文本文件

    当基于文本的文件上传到 Confluence(例如,Word,PowerPoint 等),这些文件中的文本是可以提取并且添加到索引中的,用户可以通过索引来搜索这些文件中的文本内容,不仅仅是搜索文件名. ...

  9. 论坛IP地址追踪&路由器密码嗅探

    一.论坛IP地址查询 1.任何应用程序部可以调用一个标准的库函数来查看给定名称的主机IP地址.类似地,系统还提供一个逆函致—给定主机的IP地址,查看它所对应的主机名.大多数使用主机名作为参数的应用程序 ...

  10. Nginx详解六:Nginx基础篇之Nginx日志

    1.Nginx日志类型 error.log:记录Nginx处理http请求的错误的状态,以及Nginx服务本身服务运行的错误的状态 access_log:记录通过Nginx的http请求的访问状态,用 ...