题意:
给一棵树和若干查询点对,求这些点对的lca。
分析:
tarjan求lca的模板题,树还是用孩子兄弟表示法比较简洁。
代码:
//poj 1470
//sepNINE
#include <iostream>
#include <vector>
using namespace std;
const int maxN=1024;
int n,u,v,t,m,x,y;;
int par[maxN],son[maxN],bro[maxN],f[maxN],cnt[maxN],vis[maxN];
vector<int> query[maxN];
int find(int u)
{
return f[u]==u?u:f[u]=find(f[u]);
}
void tarjan(int root)
{
int i;
f[root]=root;
vis[root]=1;
for(i=query[root].size()-1;i>=0;--i){
int p=query[root][i];
if(vis[p]==1)
++cnt[find(p)];
}
int s=son[root];
while(s){
tarjan(s);
f[s]=root;
s=bro[s];
}
}
int main()
{
char ch1[4],ch2[4],ch3[4];
int i;
while(scanf("%d",&n)==1){
memset(par,0,sizeof(par));
memset(son,0,sizeof(son));
memset(bro,0,sizeof(bro));
for(i=1;i<=n;++i) query[i].clear();
for(i=1;i<=n;++i){
scanf("%d%1s%1s%d%1s",&u,ch1,ch2,&t,&ch3);
while(t--){
scanf("%d",&v);
par[v]=1;
bro[v]=son[u];
son[u]=v;
}
}
for(i=1;i<=n;++i)
if(par[i]==0) break;
scanf("%d",&m);
while(m--){
scanf("%1s%d%d%1s",ch1,&x,&y,ch2);
query[x].push_back(y);
if(x!=y) query[y].push_back(x);
}
memset(cnt,0,sizeof(cnt));
memset(vis,0,sizeof(vis));
tarjan(i);
for(i=1;i<=n;++i)
if(cnt[i]) printf("%d:%d\n",i,cnt[i]);
}
return 0;
} poj 1470 Closest Common Ancestors tarjan求lca和树的孩子兄弟表示
原文地址:http://blog.csdn.net/sepnine/article/details/41483095