• 2049: [Sdoi2008]Cave 洞穴勘测(LCT)


    2049: [Sdoi2008]Cave 洞穴勘测

    Time Limit: 10 Sec  Memory Limit: 259 MB
    Submit: 9962  Solved: 4824
    [Submit][Status][Discuss]

    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操作以免超时

    code

     1 #include<cstdio>
     2 #include<algorithm>
     3 
     4 using namespace std;
     5 
     6 const int N = 20010;
     7 int fa[N],ch[N][2],rev[N],st[N],top;
     8 
     9 void pushdown(int x) {
    10     if (rev[x]) {
    11         rev[ch[x][0]] ^= 1;rev[ch[x][1]] ^= 1;
    12         swap(ch[x][0],ch[x][1]);
    13         rev[x] ^= 1;
    14     }
    15 }
    16 bool isroot(int x) {
    17     return ch[fa[x]][0]!=x&&ch[fa[x]][1]!=x;
    18 }
    19 int son(int x) {
    20     return ch[fa[x]][1]==x;
    21 }
    22 void rotate(int x) {
    23     int y = fa[x],z = fa[y],b = son(x),c = son(y),a = ch[x][!b];
    24     if (!isroot(y)) ch[z][c] = x;fa[x] = z;
    25     ch[x][!b] = y;fa[y] = x;
    26     ch[y][b] = a;if (a) fa[a] = y;
    27 }
    28 void splay(int x) {
    29     top = 0;st[++top] = x;
    30     for (int i=x; !isroot(i); i=fa[i]) st[++top] = fa[i];
    31     while (top) pushdown(st[top--]);
    32     while (!isroot(x)) {
    33         int y = fa[x];
    34         if (!isroot(y)) {
    35             if (son(x)==son(y)) rotate(y);
    36             else rotate(x);
    37         }
    38         rotate(x);
    39     }
    40 }
    41 void access(int x) {
    42     for (int t=0; x; t=x,x=fa[x]) {
    43         splay(x);ch[x][1] = t;
    44     }
    45 }
    46 void makeroot(int x) {
    47     access(x);
    48     splay(x);
    49     rev[x] ^= 1;
    50 }
    51 void link(int x,int y) {
    52     makeroot(x);
    53     fa[x] = y;
    54 }
    55 void cut(int x,int y) {
    56     makeroot(x);
    57     access(y);
    58     splay(y);
    59     ch[y][0] = fa[ch[y][0]] = 0;
    60 }
    61 int find(int x) {
    62     access(x);
    63     splay(x);
    64     while (ch[x][0]) x = ch[x][0];
    65     return x;
    66 }
    67 int main() {
    68     int n,m,x,y;
    69     char opt[20];
    70     scanf("%d%d",&n,&m);
    71     while (m--) {
    72         scanf("%s%d%d",opt,&x,&y);
    73         if (opt[0]=='C') link(x,y);
    74         else if (opt[0]=='D') cut(x,y);
    75         else {
    76             if (find(x)==find(y)) puts("Yes");
    77             else puts("No");
    78         }
    79     }
    80     return 0;
    81 }
  • 相关阅读:
    Passing structures between C# and C/C++
    Leetcode 题目整理-6 Swap Nodes in Pairs & Remove Duplicates from Sorted Array
    Leetcode 题目整理-5 Valid Parentheses & Merge Two Sorted Lists
    链表
    Leetcode 题目整理-4 Longest Common Prefix & Remove Nth Node From End of List
    Leetcode 题目整理-3 Palindrome Number & Roman to Integer
    Leetcode 题目整理-2 Reverse Integer && String to Integer
    Leetcode 题目整理-1
    Leetcode 题目整理
    static静态变量在c++类中的应用实例
  • 原文地址:https://www.cnblogs.com/mjtcn/p/8149977.html
Copyright © 2020-2023  润新知