• 【SDOI2008】【BZOJ2049】Cave 洞穴勘測


    Description

    辉辉热衷于洞穴勘測。某天,他依照地图来到了一片被标记为JSZX的洞穴群地区。经过初步勘測,辉辉发现这片区域由n个洞穴(分别编号为1到n)以及若干通道组成。而且每条通道连接了恰好两个洞穴。

    假如两个洞穴能够通过一条或者多条通道按一定顺序连接起来,那么这两个洞穴就是连通的,按顺序连接在一起的这些通道则被称之为这两个洞穴之间的一条路径。洞穴都十分牢固无法破坏,然而通道不太稳定,时常由于外界影响而发生改变,比方。依据有关仪器的监測结果。123号洞穴和127号洞穴之间有时会出现一条通道,有时这条通道又会由于某种稀奇古怪的原因被毁。

    辉辉有一台监測仪器能够实时将通道的每一次改变状况在辉辉手边的终端机上显示:假设监測到洞穴u和洞穴v之间出现了一条通道,终端机上会显示一条指令 Connect u v 假设监測到洞穴u和洞穴v之间的通道被毁,终端机上会显示一条指令 Destroy u v 经过长期的艰苦卓绝的手工推算。辉辉发现一个奇怪的现象:不管通道怎么改变,随意时刻随意两个洞穴之间至多仅仅有一条路径。因而,辉辉坚信这是由于某种本质规律的支配导致的。因而。辉辉更加夜以继日地坚守在终端机之前,试图通过通道的改变情况来研究这条本质规律。然而,最终有一天,辉辉在堆积成山的演算纸中崩溃了……他把终端机往地面一砸(终端机也足够牢固无法破坏),转而求助于你。说道:“你老兄把这程序写写吧”。

    辉辉希望能随时通过终端机发出指令 Query u v。向监測仪询问此时洞穴u和洞穴v是否连通。

    如今你要为他编敲代码回答每一次询问。已知在第一条指令显示之前,JSZX洞穴群中没有不论什么通道存在。

    Input

    第一行为两个正整数n和m,分别表示洞穴的个数和终端机上出现过的指令的个数。下面m行,依次表示终端机上出现的各条指令。每行开头是一个表示指令种类的字符串s(”Connect”、”Destroy”或者”Query”。区分大写和小写),之后有两个整数u和v (1≤u, v≤n且u≠v) 分别表示两个洞穴的编号。

    Output

    对每一个Query指令。输出洞穴u和洞穴v是否互相连通:是输出”Yes”,否则输出”No”。(不含双引號)

    Sample Input

    例子输入1 cave.in

    200 5

    Query 123 127

    Connect 123 127

    Query 123 127

    Destroy 127 123

    Query 123 127

    例子输入2 cave.in

    3 5

    Connect 1 2

    Connect 3 1

    Query 2 3

    Destroy 1 3

    Query 2 3

    Sample Output

    例子输出1 cave.out

    No

    Yes

    No

    例子输出2 cave.out

    Yes

    No

    HINT

    数据说明 10%的数据满足n≤1000, m≤20000 20%的数据满足n≤2000, m≤40000 30%的数据满足n≤3000, m≤60000 40%的数据满足n≤4000, m≤80000 50%的数据满足n≤5000, m≤100000 60%的数据满足n≤6000, m≤120000 70%的数据满足n≤7000, m≤140000 80%的数据满足n≤8000, m≤160000 90%的数据满足n≤9000, m≤180000 100%的数据满足n≤10000, m≤200000 保证全部Destroy指令将摧毁的是一条存在的通道本题输入、输出规模比較大,建议cc++选手使用scanf和printf进行IO操作以免超时

    Source

    动态树

    裸题啊…
    link然后cut.查询就查一下连通性…
    (请叫我写LCT必挂星人QAQ)
    写残Cut就是切不掉那条边233

    #include<iostream>
    #include<cstdio>
    #include<cstring>
    #include<cmath>
    #include<algorithm>
    #define MAXN 10010
    using namespace std;
    int n,m,u,v;
    char ch[20];
    struct splay
    {
        int fa,ch[2];
        bool rev;
    }tree[MAXN];
    int sta[MAXN],top;
    void in(int &x)
    {
        char ch=getchar();int flag=1;x=0;
        while (!(ch>='0'&&ch<='9')) flag=ch=='-'?

    -1:1,ch=getchar(); while (ch>='0'&&ch<='9') x=x*10+ch-'0',ch=getchar(); } inline bool is_root(int x) { return tree[tree[x].fa].ch[0]!=x&&tree[tree[x].fa].ch[1]!=x; } inline void push_down(int x) { if (tree[x].rev) { tree[x].rev^=1;tree[tree[x].ch[0]].rev^=1;tree[tree[x].ch[1]].rev^=1; swap(tree[x].ch[0],tree[x].ch[1]); } } inline void rot(int x) { int y=tree[x].fa,z=tree[y].fa,l,r; l=(tree[y].ch[1]==x);r=l^1; if (!is_root(y)) tree[z].ch[tree[z].ch[1]==y]=x; tree[tree[x].ch[r]].fa=y;tree[y].fa=x;tree[x].fa=z; tree[y].ch[l]=tree[x].ch[r];tree[x].ch[r]=y; } inline void Splay(int x) { sta[++top]=x; for (int i=x;!is_root(i);i=tree[i].fa) sta[++top]=tree[i].fa; while (top) push_down(sta[top--]); while (!is_root(x)) { int y=tree[x].fa,z=tree[y].fa; if (!is_root(y)) { if (tree[y].ch[0]==x^tree[z].ch[0]==y) rot(x); else rot(y); } rot(x); } } inline void access(int x) { for (int t=0;x;t=x,x=tree[x].fa) Splay(x),tree[x].ch[1]=t; } inline void make_root(int x) { access(x);Splay(x);tree[x].rev^=1; } inline void link(int x,int y) { make_root(x);tree[x].fa=y; } inline void cut(int x,int y) { make_root(x);access(y);Splay(y);tree[y].ch[0]=tree[x].fa=0; } inline int find(int x) { access(x);Splay(x); int ret=x; while (tree[ret].ch[0]) ret=tree[ret].ch[0]; return ret; } int main() { in(n);in(m); while (m--) { scanf("%s",ch);in(u);in(v); if (ch[0]=='Q') find(u)==find(v)?puts("Yes"):puts("No"); if (ch[0]=='C') link(u,v); if (ch[0]=='D') cut(u,v); } }

  • 相关阅读:
    EL 自定义函数
    Linux 软件安装管理
    Linux 网络环境查看命令
    Linux 用户和用户组的命令
    Linux 用户和用户组进阶命令
    Linux 用户和用户组的基本命令
    将博客搬至CSDN
    U盘做系统启动盘(PE)时的文件格式选择 HDD ZIP FDD
    STM32 的几种输入输出模式
    define 中强制类型转换 && 浮点数后面带f
  • 原文地址:https://www.cnblogs.com/llguanli/p/8468994.html
Copyright © 2020-2023  润新知