zoukankan      html  css  js  c++  java
  • hdu 5967 小R与手机(LCT裸题)

    题目链接:hdu 5967 小R与手机

    题意:

    有n个手机,每个手机有一个呼叫转移。如果形成了环就不能接通。

    现在有两个操作:

    1 询问打电话x,最终接到的是那个电话,如果有环,输出-1

    2 将x的呼叫转移改为y

    题解:

    用LCT来维护这个森林的关系。

    显然要形成环的话,肯定是树跟连向了该树内的点。

    所以特别维护一下树根的连向就行了。

     1 #include<bits/stdc++.h>
     2 #define F(i,a,b) for(int i=a;i<=b;i++)
     3 #define mst(a,b) memset(a,b,sizeof(a))
     4 using namespace std;
     5 
     6 namespace LCT
     7 {
     8     const int N=2e5+7;
     9     int f[N],son[N][2],tmp[N];bool rev[N];
    10     bool isroot(int x){return !f[x]||son[f[x]][0]!=x&&son[f[x]][1]!=x;}
    11     void rotate(int x){
    12         int y=f[x],w=son[y][1]==x;
    13         son[y][w]=son[x][w^1];
    14         if(son[x][w^1])f[son[x][w^1]]=y;
    15         if(f[y]){
    16             int z=f[y];
    17             if(son[z][0]==y)son[z][0]=x;else if(son[z][1]==y)son[z][1]=x;
    18         }
    19         f[x]=f[y];f[y]=x;son[x][w^1]=y;
    20     }
    21     void splay(int x){
    22         int s=1,i=x,y;tmp[1]=i;
    23         while(!isroot(x)){
    24             y=f[x];
    25             if(!isroot(y)){if((son[f[y]][0]==y)^(son[y][0]==x))rotate(x);else rotate(y);}
    26             rotate(x);
    27         }
    28     }
    29     void access(int x){for(int y=0;x;y=x,x=f[x])splay(x),son[x][1]=y;}
    30     int root(int x){access(x);splay(x);while(son[x][0])x=son[x][0];return x;}
    31     void makeroot(int x){access(x);splay(x);}
    32     void link(int x,int y){makeroot(x);f[x]=y;access(x);}
    33     void cutf(int x){access(x);splay(x);f[son[x][0]]=0;son[x][0]=0;}
    34     void cut(int x,int y){makeroot(x);cutf(y);}
    35 }
    36 
    37 using namespace LCT;
    38 int n,m,x,y,op,fa[N],a[N];
    39 
    40 void Link(int x,int y)
    41 {
    42     int rt=root(x);
    43     if(rt==x)fa[x]=0;else cutf(x);
    44     if(fa[rt]&&root(fa[rt])!=rt)
    45         link(rt,fa[rt]),fa[rt]=0;
    46     if(!y)return;
    47     if(root(y)==x)fa[x]=y;else link(x,y);
    48 }
    49 
    50 int main()
    51 {
    52     scanf("%d%d",&n,&m);
    53     F(i,1,n)
    54     {
    55         scanf("%d",a+i);
    56         if(a[i])Link(i,a[i]);
    57     }
    58     F(i,1,m)
    59     {
    60         scanf("%d",&op);
    61         if(op==1)scanf("%d%d",&x,&y),Link(x,y);
    62         else
    63         {
    64             scanf("%d",&x);
    65             int rt=root(x);
    66             if(fa[rt])puts("-1");
    67             else printf("%d
    ",rt);
    68         }
    69     }
    70     return 0;
    71 }
    View Code
  • 相关阅读:
    架构设计中的方法学——迭代设计
    架构重构改善既有代码的设计
    架构如何为业务和技术“服务”(2)
    架构如何为业务和技术“服务”(1)
    选用面向领域的多层分布式架构(DDD风格架构)的理由
    传话游戏(2013编程之美全国挑战赛资格赛)
    2012年蓝桥杯软件设计大赛河南省初赛试题和答案
    最大值最下化
    NYOJ120 校园网络(强连通缩点targan算法)
    NYOJ176整数划分(二)
  • 原文地址:https://www.cnblogs.com/bin-gege/p/7696033.html
Copyright © 2011-2022 走看看