zoukankan      html  css  js  c++  java
  • POJ 1470 -- Closest Common Ancestors

    Closest Common Ancestors
    Time Limit: 2000MS   Memory Limit: 10000K
    Total Submissions: 21483   Accepted: 6812

    Description

    Write a program that takes as input a rooted tree and a list of pairs of vertices. For each pair (u,v) the program determines the closest common ancestor of u and v in the tree. The closest common ancestor of two nodes u and v is the node w that is an ancestor of both u and v and has the greatest depth in the tree. A node can be its own ancestor (for example in Figure 1 the ancestors of node 2 are 2 and 5)

    Input

    The data set, which is read from a the std input, starts with the tree description, in the form: 

    nr_of_vertices 
    vertex:(nr_of_successors) successor1 successor2 ... successorn 
    ...
    where vertices are represented as integers from 1 to n ( n <= 900 ). The tree description is followed by a list of pairs of vertices, in the form: 
    nr_of_pairs 
    (u v) (x y) ... 

    The input file contents several data sets (at least one). 
    Note that white-spaces (tabs, spaces and line breaks) can be used freely in the input.

    Output

    For each common ancestor the program prints the ancestor and the number of pair for which it is an ancestor. The results are printed on the standard output on separate lines, in to the ascending order of the vertices, in the format: ancestor:times 
    For example, for the following tree: 

    Sample Input

    5
    5:(3) 1 4 2
    1:(0)
    4:(0)
    2:(1) 3
    3:(0)
    6
    (1 5) (1 4) (4 2)
          (2 3)
    (1 3) (4 3)

    Sample Output

    2:1
    5:5

    Hint

    Huge input, scanf is recommended.
     
     
    题意:给出顶点数,各顶点的子节点数目和编号,询问次数,每次询问的顶点对,要求输出询问中出现的最近公共祖先和与其对应的询问次数;
    题解&总结:
    1. LCA裸题,下面给出Tarjan离线解法的代码,Tarjan算法网上已有许多不错的讲解,在此不再赘述,把dfs和并查集理解好Tarjan算法也就挺好理解了;
    2. 输入有多组数据,注意把数组和变量初始化;
    3. 输入的第一个数是节点数,不一定是根节点,根节点的确定应该通过排除子节点,剩下的一个就是根节点;
    4. 代码中以前向星来储存边,当然用vector来实现邻接表也可以,但显然前向星比vector更快;
    5. 题目中只给出了最大节点数,但并没有给出最大询问次数,经多次提交实验,最大询问次数应该在250000左右,注意开足够大的数组;
    6. 通过此题可以发现,数组开小了不止可能会RE,还可能会访问到垃圾数据使dfs无穷递归而MLE,也可能出现死循环而TLE;
    7. 题目提示推荐用scanf输入,本着倔强的精神试了cin输入也还是可以1900+MS过了。
     
     1 #include <iostream>
     2 #include <stdio.h>
     3 #include <cstring>
     4 #define N 1010
     5 #define MAXQ 250010
     6 using namespace std;
     7 struct node{
     8     int next, to, lca;
     9 } edge[2*N], qedge[2*MAXQ];  // 储存边,依次储存询问
    10 int num_edge, num_qedge, head[N], qhead[N];
    11 int fa[N];
    12 bool vis[N], flag[N];
    13 void add_edge(int from, int to) { // 前向星添加边
    14     edge[++num_edge].next = head[from];
    15     edge[num_edge].to = to;
    16     head[from] = num_edge;
    17 }
    18 void add_qedge(int from, int to) {
    19     qedge[++num_qedge].next = qhead[from];
    20     qedge[num_qedge].to = to;
    21     qhead[from] = num_qedge;
    22 }
    23 int Find(int x) {
    24     if (fa[x] != x) fa[x] = Find(fa[x]);
    25     return fa[x];
    26 }
    27 void dfs(int x) {
    28     fa[x] = x;
    29     vis[x] = true;
    30     for (int k = head[x]; k; k = edge[k].next) {
    31         if (!vis[edge[k].to]) {
    32             dfs(edge[k].to);
    33             fa[edge[k].to] = x;
    34         }
    35     }
    36     for (int k = qhead[x]; k; k = qedge[k].next) {
    37         if (vis[qedge[k].to]) {
    38             qedge[k].lca = Find(qedge[k].to);
    39             if (k & 1) qedge[k+1].lca = qedge[k].lca;
    40             else qedge[k-1].lca = qedge[k].lca;
    41         }
    42     }
    43 }
    44 void init() {  // 初始化
    45     memset(vis, false, sizeof(vis));
    46     memset(flag, false, sizeof(flag));
    47     memset(head, 0, sizeof(head));
    48     memset(qhead, 0, sizeof(qhead));
    49     num_edge = num_qedge = 0;
    50 }
    51 
    52 int main() {
    53     int n, x, y, k;
    54     char ch;
    55     ios::sync_with_stdio(false); cin.tie(0); cout.tie(0);
    56     while (cin >> n) {
    57         init();
    58         for (int i = 0; i < n; i++) {
    59             cin >> x >> ch >> ch >> k >> ch;//scanf("%d%*c%*c%d%*c", &x, &k);
    60             while (k--) {
    61                 cin >> y;//scanf(" %d", &y);
    62                 flag[y] = true;
    63                 add_edge(x, y); add_edge(y, x);
    64             }
    65 
    66         }
    67         cin >> k;//scanf("%d", &k);
    68         for (int i = 0; i < k; i++) {
    69             cin >> ch >> x >> y >> ch;//scanf(" %*c%d %d%*c", &x, &y);
    70             add_qedge(x, y); add_qedge(y, x);
    71         }
    72         for (int i = 1; i <= n; i++) if (!flag[i]) dfs(i);  // 确定根节点
    73         int cnt[N]; memset(cnt, 0, sizeof(cnt));
    74         for (int i = 1; i <= k; i++) cnt[qedge[i*2].lca]++;
    75         for (int i = 1; i <= n; i++) if (cnt[i]) cout << i << ':' << cnt[i] << endl;//printf("%d:%d
    ", i, cnt[i]);
    76     }
    77 
    78     return 0;
    79 }
    View Code
     
    作者:_kangkang
    本文版权归作者和博客园共有,欢迎转载,但必须给出原文链接,并保留此段声明,否则保留追究法律责任的权利。
  • 相关阅读:
    ubuntu开启SSH服务
    [FreeModbus源码分析] 1.协议简介
    minicom无法输入问题
    Redis学习sorted set数据类型
    Redis学习string数据类型
    Redis学习list数据类型
    Redis学习常用命令
    Redis学习hash数据类型
    Redis学习set数据结构
    Redis初探Redis安装
  • 原文地址:https://www.cnblogs.com/kangkang-/p/8398760.html
Copyright © 2011-2022 走看看