POJ_1330 Nearest Common Ancestors LCA

题目链接:http://poj.org/problem?id=1330

最近公共祖先模板题

Tarjan离线算法 :

读取所有询问后保存并按照一定方式排列。该算法主要采取深度优先搜索的方式,结合并查集,在搜索同时处理查询,大致过程如下:

搜索到节点u

  访问标记u

  对于u的所有子节点递归搜索,每搜索并处理完一个子节点后将其与当前节点合并,u为父节点

  若有一个与u关联的询问 q(u, v) 并且 v 已经被访问过,则其lca为 find(v) - 即并查集父节点函数

为何lca为 find(v) :

如果有一个从当前结点到结点v的询问,且v已被检查过,则由于进行的是深度优先搜索,当前结点与v的最近公共祖先一定还没有被检查,而这个最近公共祖先的包涵v的子树一定已经搜索过了,那么这个最近公共祖先一定是v所在集合的祖先。

参考:http://dongxicheng.org/structure/lca-rmq/

实际代码编写中需要注意初始化,并查集合并时父节点是否一定等问题,细节处理要谨慎。

伪代码:

 1 lca(node u){
 2     fa[u] = u;
 3     vis[u] = 1;
 4     for every son of u{
 5         lca(u.son)
 6         union(u, u.son);
 7     }
 8     for every query of u {
 9         if vis[q[u]] == 1
10             ans[u].push_back(find(q[u]))
11     }
12 }

题目AC代码:

 1 const int inf = 0x3f3f3f3f;
 2 const int maxn = 1e4 + 5;
 3 struct Edge{
 4     int to, next, val;
 5 };
 6 Edge edges[maxn * 2];
 7 int tot, head[maxn], vis[maxn];
 8 int n;
 9 int fa[maxn];
10 int qu, qv, ans;
11 int deg[maxn];
12
13 void init(){
14     memset(deg, 0, sizeof(deg));
15     memset(edges, 0, sizeof(edges));
16     memset(head, -1, sizeof(head));
17     memset(vis, 0, sizeof(vis));
18     for(int i = 0; i <= n; i++) fa[i] = i;
19     tot = 0;
20 }
21 void addEdge(int u, int v, int w){
22     edges[tot].to = v;
23     edges[tot].val = w;
24     edges[tot].next = head[u];
25     head[u] = tot++;
26 }
27
28 int find(int x){
29     return (fa[x] == x)? x: fa[x] = find(fa[x]);
30 }
31 void unite(int x, int y){
32     int fx = find(x), fy = find(y);
33     if(fx == fy) return;
34     fa[fy] = fx;
35 }
36
37 void lca(int u){
38     fa[u] = u;
39     vis[u] = 1;
40     for(int i = head[u]; i != -1; i = edges[i].next){
41         if(!vis[edges[i].to]) {
42             lca(edges[i].to);
43             unite(u, edges[i].to);
44         }
45     }
46     if(u == qu && vis[qv] == 1){
47         ans = find(qv);
48     }
49     if(u == qv && vis[qu] == 1){
50         ans = find(qu);
51     }
52 }
53
54
55 int main(){
56     int T;
57     scanf("%d", &T);
58     while(T--){
59         scanf("%d", &n);
60         init();
61         for(int i = 0; i < n - 1; i++){
62             int from, to;
63             scanf("%d %d", &from, &to);
64             deg[to]++;
65             addEdge(from, to, 1);
66             addEdge(to, from, 1);
67         }
68         int root = 1;
69         for(int i = 1; i <= n; i++) if(deg[i] == 0) root = i;
70         scanf("%d %d", &qu, &qv);
71         lca(root);
72         printf("%d\n", ans);
73     }
74 }

题目:

earest Common Ancestors

Time Limit: 1000MS   Memory Limit: 10000K
Total Submissions: 29509   Accepted: 15069

Description

A rooted tree is a well-known data structure in computer science and engineering. An example is shown below:

 
In the figure, each node is labeled with an integer from {1, 2,...,16}. Node 8 is the root of the tree. Node x is an ancestor of node y if node x is in the path between the root and node y. For example, node 4 is an ancestor of node 16. Node 10 is also an ancestor of node 16. As a matter of fact, nodes 8, 4, 10, and 16 are the ancestors of node 16. Remember that a node is an ancestor of itself. Nodes 8, 4, 6, and 7 are the ancestors of node 7. A node x is called a common ancestor of two different nodes y and z if node x is an ancestor of node y and an ancestor of node z. Thus, nodes 8 and 4 are the common ancestors of nodes 16 and 7. A node x is called the nearest common ancestor of nodes y and z if x is a common ancestor of y and z and nearest to y and z among their common ancestors. Hence, the nearest common ancestor of nodes 16 and 7 is node 4. Node 4 is nearer to nodes 16 and 7 than node 8 is.

For other examples, the nearest common ancestor of nodes 2 and 3 is node 10, the nearest common ancestor of nodes 6 and 13 is node 8, and the nearest common ancestor of nodes 4 and 12 is node 4. In the last example, if y is an ancestor of z, then the nearest common ancestor of y and z is y.

Write a program that finds the nearest common ancestor of two distinct nodes in a tree.

Input

The input consists of T test cases. The number of test cases (T) is given in the first line of the input file. Each test case starts with a line containing an integer N , the number of nodes in a tree, 2<=N<=10,000. The nodes are labeled with integers 1, 2,..., N. Each of the next N -1 lines contains a pair of integers that represent an edge --the first integer is the parent node of the second integer. Note that a tree with N nodes has exactly N - 1 edges. The last line of each test case contains two distinct integers whose nearest common ancestor is to be computed.

Output

Print exactly one line for each test case. The line should contain the integer that is the nearest common ancestor.

Sample Input

2
16
1 14
8 5
10 16
5 9
4 6
8 4
4 10
1 13
6 15
10 11
6 7
10 2
16 3
8 1
16 12
16 7
5
2 3
3 4
3 1
1 5
3 5

Sample Output

4
3
时间: 2024-10-10 04:20:47

POJ_1330 Nearest Common Ancestors LCA的相关文章

POJ 1330 Nearest Common Ancestors LCA题解

本题是一个多叉树,然后求两点的最近公共单亲节点. 就是典型的LCA问题.这是一个很多解法的,而且被研究的很透彻的问题. 原始的解法:从根节点往下搜索,若果搜索到两个节点分别在一个节点的两边,那么这个点就是最近公共单亲节点了. Trajan离线算法:首次找到两个节点的时候,如果记录了他们的最低单亲节点,那么答案就是这个最低的单亲节点了. 问题是如何有效记录这个最低单亲节点,并有效根据遍历的情况更新,这就是利用Union Find(并查集)记录已经找到的节点,并及时更新最新访问的节点的当前最低单亲节

POJ 1330 Nearest Common Ancestors LCA(在线RMQ,离线Tarjan)

链接:http://poj.org/problem?id=1330 题意:只看题目就知道题目是什么意思了,最近公共祖先,求在一棵树上两个节点的最近公共祖先. 思路:求最近公共祖先有两种算法,在线和离线,在线方法是用RMQ求LCA,一句话总结就是在从DFS时,从第一个点到第二个点的最短路径中深度最浅的点就是公共祖先,用RMQ处理,一般问题的最优解决方式的复杂度是O(NlogN)的预处理+N*O(1)的查询.离线方法是Tarjan算法,将所有询问的两个点都记录下来,在DFS过程中不断将每个点自身作为

poj 1330 Nearest Common Ancestors lca 在线rmq

Nearest Common Ancestors Description A rooted tree is a well-known data structure in computer science and engineering. An example is shown below:  In the figure, each node is labeled with an integer from {1, 2,...,16}. Node 8 is the root of the tree.

POJ 1330 Nearest Common Ancestors LCA

题目链接: http://poj.org/problem?id=1330 题意: 给你一颗有根树,最后输入一对数(a,b),叫你求a和b的公共祖先. 裸的lca,数据也很小,拿来练手不错. 题解: 1.tarjan_lca,离线,线性时间复杂度 代码: 1 #include<iostream> 2 #include<cstdio> 3 #include<cstring> 4 #include<vector> 5 using namespace std; 6

POJ 1330 Nearest Common Ancestors(LCA模板)

给定一棵树求任意两个节点的公共祖先 tarjan离线求LCA思想是,先把所有的查询保存起来,然后dfs一遍树的时候在判断.如果当前节点是要求的两个节点当中的一个,那么再判断另外一个是否已经访问过,如果访问过的话,那么它的最近公共祖先就是当前节点祖先. 下面是tarjan离线模板: #include <cstdio> #include <cstring> #include <algorithm> using namespace std; const int maxn =

[POJ1330]Nearest Common Ancestors(LCA, 离线tarjan)

题目链接:http://poj.org/problem?id=1330 题意就是求一组最近公共祖先,昨晚学了离线tarjan,今天来实现一下. 个人感觉tarjan算法是利用了dfs序和节点深度的关系,大致的意思:dfs如果不递归到递归基,那么dfs就会越递归越深,这个时候深度也是相应增加的,所以这个时候任意在已经遍历过的节点中选取两个点,计算他们的lca也就相当于是用并查集求他们的root.而dfs执行到递归基,转而执行下一个分支的时候,这个时候dfs的节点应当是小于等于之前执行到递归基的节点

POJ 1330 Nearest Common Ancestors 倍增算法的LCA

POJ 1330 Nearest Common Ancestors 题意:最近公共祖先的裸题 思路:LCA和ST我们已经很熟悉了,但是这里的f[i][j]却有相似却又不同的含义.f[i][j]表示i节点的第2j个父亲是多少   这个代码不是我的,转自 邝斌博客 1 /* *********************************************** 2 Author :kuangbin 3 Created Time :2013-9-5 9:45:17 4 File Name :F

poj 1330 Nearest Common Ancestors 【LCA】

Nearest Common Ancestors Time Limit: 1000MS   Memory Limit: 10000K Total Submissions: 20073   Accepted: 10631 Description A rooted tree is a well-known data structure in computer science and engineering. An example is shown below: In the figure, each

poj 1330 Nearest Common Ancestors 单次LCA/DFS

Nearest Common Ancestors Time Limit: 1000MS   Memory Limit: 10000K Total Submissions: 19919   Accepted: 10544 Description A rooted tree is a well-known data structure in computer science and engineering. An example is shown below: In the figure, each