https://vjudge.net/problem/UVA-12219
思路:把每一个子树用一个数代替放到map里,这样查一棵子树是否出现就是快多了。递归打印答案即可。
#include<bits/stdc++.h> #define _for(i,a,b) for(int i=a;i<=b;i++) using namespace std; typedef long long ll; const int mod =1e6+7; double esp=1e-6; int INF =0x3f3f3f3f; const int inf = 1<<28; const int MAXN=60000; char s[MAXN*5],*p; int T,n,m,cnt,kase; struct node { string s; int hash,l,r; bool operator <(const node&p)const//map比较的条件 { if(hash!=p.hash)return hash<p.hash; if(l!=p.l)return l<p.l; return r<p.r; } }nodes[MAXN]; map<node,int> mp; int solve() { int id=cnt++; node& u=nodes[id]; u.l=u.r=-1; u.s=""; u.hash=0; while(isalpha(*p)) { u.hash=u.hash*27+*p-'a'+1; u.s+=*p; ++p; } if(p&&(*p)=='(') { p++; u.l=solve(); p++; u.r=solve(); p++; } if(mp.count(u))//如果map里有,就直接返回这个子树的id { cnt--; return mp[u]; } return mp[u]=id; } int ex[MAXN]; void print(int v)//递归打印 { if(ex[v]!=kase) { ex[v]=kase; cout<<nodes[v].s; if(nodes[v].l!=-1) { printf("("); print(nodes[v].l); printf(","); print(nodes[v].r); printf(")"); } } else { printf("%d",v); } } int main() { scanf("%d",&T); for(kase=1;kase<=T;kase++) { mp.clear(); scanf("%s",s); p=s; cnt=1; int tmp=solve(); print(tmp); printf("\n"); } return 0; }