開始的時候只是用了tire樹。。但是沒有判斷圖是否連通。。。覺得只要奇數個的是0或2就好了。。後來發現了這樣的情況:b b\n s r\n s t\n滿足奇數個那個條件但是是impossible的。。於是用並查集。。想原來最開始學習的就是並查集,現在竟然很是生疏。。。union這個函數都用錯了。。。=。=路漫漫啊~
上代碼:
#include<cstdio>
#include<cstring>
using namespace std;
struct node{
node *next[27];
int cnt;
int id;
node(){
int i;
cnt=1;
id=-1;
for(i=0;i<=25;i++){
next[i]=NULL;
}
}
};
int odd=0,even=0;
int num=1;
int insert(node* &root,char *word){
node *cur=root;
int i=0,branch;
if(cur==NULL){
cur=new node();
root=cur;
}
while(word[i]){
branch=word[i]-'a';
if(cur->next[branch]){
cur->next[branch]->cnt++;
}
else{
cur->next[branch]=new node();
}
i++;
cur=cur->next[branch];
}
if(cur->id==-1){
cur->id=num++;
}
if((cur->cnt)%2==0){
odd--;
even++;
}
else{
if(cur->cnt==1){
odd++;
}
else{
even--;
odd++;
}
}
return cur->id;
}
int f[510010];
void set(){//初始化
int i;
for(i=0;i<=500008;i++){
f[i]=i;
}
}
int find(int i){//找朋友
if(f[i]!=i)
f[i]=find(f[i]);
return f[i];
}
void unoin(int rx,int ry){//連接朋友
f[rx]=ry;
}
main(){
char ch[50],a[20],b[20];
node *root=NULL;
int k=0,i,j;
set();
while(gets(ch)){
if(strcmp("",ch)==0)
break;
sscanf(ch,"%s%s",a,b);
i=insert(root,a);
j=insert(root,b);
if(i<j)
unoin(find(i),find(j);
else
unoin(find(j),find(i));
}
if(odd%2==0 && odd<=2){
int flag=1;
k=find(1);
for(i=2;i<=num-1;i++){
if(find(i)!=k){
flag=0;
//break;
}
}
if(flag)
printf("Possible\n");
else{
printf("Impossible\n");
}
}
else{
printf("Impossible\n");
}
}