原题干(由于是权限题我就直接砸出原题干了,要看题意概述的话在下面):

Description

黑客们通过对已有的病毒反编译,将许多不同的病毒重组,并重新编译出了新型的重组病毒。这种病毒的繁殖和变异能力极强。为了阻止这种病毒传播,某安全机构策划了一次实验,来研究这种病毒。
实验在一个封闭的局域网内进行。局域网内有n台计算机,编号为1~n。一些计算机之间通过网线直接相连,形成树形的结构。局域网中有一台特殊的计算机,称之为核心计算机。根据一些初步的研究,研究员们拟定了一个一共m步的实验。实验开始之前,核心计算机的编号为1,每台计算机中都有病毒的一个变种,而且每台计算机中的变种都不相同。实验中的每一步会是下面中的一种操作:
1、 RELEASE x
在编号为x的计算机中植入病毒的一个新变种。这个变种在植入之前不存在于局域网中。
2、 RECENTER x
将核心计算机改为编号为x的计算机。但是这个操作会导致原来核心计算机中的病毒产生新变种,并感染过来。换言之,假设操作前的核心计算机编号为y,相当于在操作后附加了一次RELEASE y的操作。
根据研究的结论,在植入一个新变种时,病毒会在局域网中搜索核心计算机的位置,并沿着网络中最短的路径感染过去。
而第一轮实验揭露了一个惊人的真相:病毒的不同变种是互斥的。新变种在感染一台已经被旧变种感染的电脑时,会把旧变种完全销毁之后再感染。但研究员发现了实现过程中的漏洞。如果新变种在感染过程中尚未销毁过这类旧变种,需要先花费1单位时间分析旧变种,才能销毁。如果之前销毁过这类旧变种,就可以认为销毁不花费时间。病毒在两台计算机之间的传播亦可认为不花费时间。
研究员对整个感染过程的耗时特别感兴趣,因为这是消灭病毒的最好时机。于是在m步实验之中,研究员有时还会做出如下的询问:
3、 REQUEST x
询问如果在编号为x的计算机的关键集合中的计算机中植入一个新变种,平均感染时间为多长。编号为y的计算机在编号为x的计算机的关键集合中,当且仅当从y沿网络中的最短路径感染到核心计算机必须经过x。由于有RECENTER操作的存在,这个集合并不一定是始终不变的。
至此,安全机构认为已经不需要实际的实验了,于是他们拜托你编写一个程序,模拟实验的结果,并回答所有的询问。

Input

输入的第一行包含两个整数n和m,分别代表局域网中计算机的数量,以及操作和询问的总数。
接下来n-1行,每行包含两个整数x和y,表示局域网中编号为x和y的计算机之间有网线直接相连。
接下来m行,每行包含一个操作或者询问,格式如问题描述中所述。

Output

对于每个询问,输出一个实数,代表平均感染时间。输出与答案的绝对误差不超过 10^(-6)时才会被视为正确。

Sample Input

8 6
1 2
1 3
2 8
3 4
3 5
3 6
4 7
REQUEST 7
RELEASE 3
REQUEST 3
RECENTER 5
RELEASE 2
REQUEST 1

Sample Output

4.0000000000
2.0000000000
1.3333333333

HINT

N<=100000,M<=100000.

题意概述:给出一棵树,初始每个结点有不同的颜色。现在支持三种操作:1.把某个结点的颜色改成一个之前都没有出现过的颜色,并将这个点到当前树根路径上的所有点全部改成这个颜色;2.改变当前的树根到另外一个点,并对原来的树根进行一次操作1;3.把询问当前形态的树中对一个点的子树中所有点进行操作1的平均代价(操作1代价的计算方式:这个点到当前树根路径上的不同颜色数量)。

从原题干来分析,相同的颜色一定连续出现在一条链上。再仔细想想,这样的操作正是LCT中的access操作!而每一次操作的代价就是access的时候经历的虚边的数量。如果是单点询问的话这就是一道LCT的裸题,但是问题在于题目要求的是子树询问。对子树信息的维护很容易想到用DFS序配合线段树来搞定。(于是就有两种流派:线段树单独在外面维护子树信息和就在LCT中维护子树信息两种,我选择线段树,常数小啊!如果要Link和Cut的话大不了我再来个LCT维护一下DFS序。。。)

分析没有换根操作的情况,在access操作中,令函数调用的时候的对象为s,当前所在结点为x,上一次所在的splay的根结点为y。y所在的splay就是s->y所在splay权值(真实树中深度)最小的点的一条链。我们找到链顶,可以发现这个链顶的子树中所有及结点在这一次access跳跃之后答案都要-1(每个点到根上的路径都少了一种颜色),x在当前树中位于splay维护的链上的儿子的子树中所有结点的答案+1。一路维护上去所有被影响的点的答案都被更新了。

换根的情况?可以发现在没有换根操作的时候可以保证对每个点的子树进行操作的时候一定是对应dfs序中一段连续的序列,但是换根之后就可以出现当前某个点子树在以1为根的dfs序中序列不连续,但是实际上也可以发现就是被分成了两端,x的子树对应的dfs序就是把x当前的父亲从dfs序中挖掉剩下的部分。对于是不是要挖点的判断借助DFS序就可以做到了。

随便分析几下思路就有了,也不是很难码,关键是细节,各种情况考虑全(为了节省您的时间请一定把一切想清楚了再开始码)。。。。。。这里只说一点:注意对LCT每个点的儿子的正确维护!

 #include<iostream>
#include<cstdio>
#include<cstring>
#include<cstdlib>
#include<algorithm>
#include<cmath>
#include<queue>
#include<set>
#include<map>
#include<vector>
#include<cctype>
using namespace std;
const int MAXN=;
typedef long long LL; int N,M,root_now;
struct edge{ int to,next; }E[MAXN<<];
int first[MAXN],np,val[MAXN],dfs_clock,l[MAXN],r[MAXN];
struct segment_tree{
static const int maxn=;
int rt,np,lc[maxn],rc[maxn]; LL sum[maxn],add[maxn];
void pushup(int now) { sum[now]=sum[lc[now]]+sum[rc[now]]; }
void pushdown(int now,int L,int R)
{
if(!add[now]) return;
int m=L+R>>;
add[lc[now]]+=add[now],sum[lc[now]]+=add[now]*(m-L+);
add[rc[now]]+=add[now],sum[rc[now]]+=add[now]*(R-m);
add[now]=;
}
void build(int &now,int L,int R,int *a)
{
now=++np;
lc[now]=rc[now]=sum[now]=add[now]=;
if(L==R){ sum[now]=a[L]; return;}
int m=L+R>>;
build(lc[now],L,m,a); build(rc[now],m+,R,a);
pushup(now);
}
void update(int now,int L,int R,int A,int B,int v)
{
if(A<=L&&R<=B){
add[now]+=v,sum[now]+=(R-L+)*v;
return;
}
pushdown(now,L,R);
int m=L+R>>;
if(B<=m) update(lc[now],L,m,A,B,v);
else if(A>m) update(rc[now],m+,R,A,B,v);
else update(lc[now],L,m,A,B,v),update(rc[now],m+,R,A,B,v);
pushup(now);
}
LL query(int now,int L,int R,int A,int B)
{
if(A<=L&&R<=B) return sum[now];
pushdown(now,L,R);
int m=L+R>>;
if(B<=m) return query(lc[now],L,m,A,B);
if(A>m) return query(rc[now],m+,R,A,B);
return query(lc[now],L,m,A,B)+query(rc[now],m+,R,A,B);
}
}st;
struct link_cut_tree{
static const int maxn=;
struct node{
int fa,ch[]; bool rev;
node(){ fa=ch[]=ch[]=,rev=; }
}nd[maxn];
void link(int x,int d,int y) { nd[x].ch[d]=y,nd[y].fa=x; }
bool isrt(int x) { return nd[nd[x].fa].ch[]!=x&&nd[nd[x].fa].ch[]!=x; }
void pushdown(int x)
{
if(!nd[x].rev) return;
int lc=nd[x].ch[],rc=nd[x].ch[];
if(lc) nd[lc].rev^=,swap(nd[lc].ch[],nd[lc].ch[]);
if(rc) nd[rc].rev^=,swap(nd[rc].ch[],nd[rc].ch[]);
nd[x].rev=;
}
void rot(int x)
{
int y=nd[x].fa,z=nd[y].fa;
pushdown(y);pushdown(x);
int d=nd[y].ch[]==x;
if(!isrt(y)) link(z,y==nd[z].ch[],x); nd[x].fa=z;
link(y,d^,nd[x].ch[d]);
link(x,d,y);
}
void splay(int x)
{
pushdown(x);
while(!isrt(x)){
int y=nd[x].fa,z=nd[y].fa;
if(!isrt(y)) rot((x==nd[y].ch[])==(y==nd[z].ch[])?y:x);
rot(x);
}
}
int find(int x,int d)
{
while(nd[x].ch[d]){ pushdown(x),x=nd[x].ch[d]; }
return x;
}
void access(int x)
{
int y=,xx,yy;
while(x){
splay(x);
if(y){
yy=find(y,);
if(l[x]<l[yy]&&l[yy]<=r[x]) st.update(st.rt,,N,l[yy],r[yy],-);
else{
if(l[x]>) st.update(st.rt,,N,,l[x]-,-);
if(r[x]<N) st.update(st.rt,,N,r[x]+,N,-);
}
}
pushdown(x);
if((xx=find(nd[x].ch[],))&&l[x]<l[xx]&&l[xx]<=r[x]) st.update(st.rt,,N,l[xx],r[xx],);
else if(xx){
if(l[x]>) st.update(st.rt,,N,,l[x]-,);
if(r[x]<N) st.update(st.rt,,N,r[x]+,N,);
}
nd[x].ch[]=y,y=x,x=nd[x].fa;
}
}
void mroot(int x)
{
access(x); splay(x);
nd[x].rev^=,swap(nd[x].ch[],nd[x].ch[]);
}
}lct; void add_edge(int u,int v)
{
E[++np]=(edge){v,first[u]};
first[u]=np;
}
void data_in()
{
scanf("%d%d",&N,&M);
int x,y;
for(int i=;i<N;i++){
scanf("%d%d",&x,&y);
add_edge(x,y); add_edge(y,x);
}
}
void DFS(int i,int f,int d)
{
l[i]=++dfs_clock,val[l[i]]=d;
for(int p=first[i];p;p=E[p].next){
int j=E[p].to;
if(j==f) continue;
lct.nd[j].fa=i;
DFS(j,i,d+);
}
r[i]=dfs_clock;
}
void work()
{
DFS(root_now=,,);
st.build(st.rt,,N,val);
char op[]; int x,len; double ans;
for(int i=;i<=M;i++){
scanf("%s%d",op,&x);
if(op[]=='L') lct.access(x);
else if(op[]=='C') lct.mroot(root_now=x);
else if(op[]=='Q'){
ans=,len=;
if(l[x]<=l[root_now]&&r[root_now]<=r[x]){
lct.splay(x);
x=lct.nd[x].ch[]?lct.find(lct.nd[x].ch[],):lct.nd[x].fa;
if(l[x]>) ans+=1.0*st.query(st.rt,,N,,l[x]-),len+=l[x]-;
if(r[x]<N) ans+=1.0*st.query(st.rt,,N,r[x]+,N),len+=N-r[x];
ans/=len;
}
else ans=1.0*st.query(st.rt,,N,l[x],r[x])/(r[x]-l[x]+);
printf("%.10f\n",ans);
}
}
}
int main()
{
freopen("test.in","r",stdin);
freopen("test.out","w",stdout);
data_in();
work();
return ;
}

最新文章

  1. Cross-Entropy Loss 与Accuracy的数值关系
  2. Win10 for Phone 裁剪控件
  3. WPF中ComboBox绑定数据库自动读取产生数据
  4. UIMenuController的使用,对UILabel拷贝以及定制菜单
  5. Andorid开发学习---ubuntu 12.04下搭建超好用的安卓模拟器genymotion 安装卸载virtualbox 4.3
  6. URAL
  7. ImageView 设置图片
  8. 北京时间28号0点以后Scott Hanselman同志台宣布dotnet core 1.0 rtm
  9. C++ static内容小结
  10. SQL2012之FileTable与C#的联合应用
  11. autoprefixer安装或者里sass的$mixin处理浏览器前缀
  12. delphi 网页提交按钮执行点击事件
  13. hdu 5524 二叉树找规律,二进制相关
  14. ubunut系统清理系统根目录下缓存文件夹.cache超大导致磁盘不足
  15. jsp基础语言-jsp声明
  16. 【BZOJ3451】Normal (点分治)
  17. 一个优秀windows C++ 程序员该有哪些知识
  18. python之__dict__与dir(转载)
  19. js简单验证码的生成和验证
  20. Linux CentOS7系统中ssh的用法

热门文章

  1. AngularJS表格排序
  2. iOS之出现( linker command failed with exit code 1)错误总结
  3. 2、开发环境搭建-window平台
  4. ABAP术语-Customer Enhancement
  5. 全盘解决eclipse之maven项目报错
  6. sqlachemy详解
  7. ASP.NET安全验证
  8. filter-policy和AS-PATH-FILTER过滤BGP路由条目
  9. QQ群认证 人数再度扩容 权限随之升级
  10. rhel6.4扩充swap分区