AC通道:http://codevs.cn/problem/1051/
[分析]
法一:
看到这道题目,我的第一感觉就是字典树。将输入的每个字符串构建一棵字典树,对于每一个叶子结点,统计以它到根的路径上的节点为末尾的单词的个数,就是它能够接出的最长的龙。结果我就把自己坑了。因为题目所给的空间只有128M,而字典树所需要的空间比较大,所以这种解法是不可行的(至少我打不出来)。可怜我debug一个上午来省空间啊!
80分代码:
#include <iostream>
#include <cstdio>
#include <cstring>
#include <vector>
#include <string>
using namespace std;
int n,sssss=0,maxn=0,cnt;
string s[100001];
struct ss{
char c;
int nxt;
ss(char a=0,int b=-1):c(a),nxt(b){}
};
struct Node{
vector<ss>child;
bool yes;
Node(){yes=false;}
};
int havecheck(vector<ss>a,char b){
if(a.empty())return false;
for(vector<ss>::iterator it=a.begin();it!=a.end();it++){
if((*it).c==b)return (*it).nxt;
}
return false;
}
vector<Node>trie;
void make_tree(int x){
// printf("%s\n",s[x]);
int l=(s[x]).size(),now=0,tmp;
if(tmp=havecheck(trie[now].child,s[x][0])){
now=tmp;
}
else{
Node a;
trie.push_back(a);
sssss++;
// trie[now].child[s[x][0]-'a']=++sssss;
(trie[now].child).push_back(ss(s[x][0],sssss));
now=sssss;
}
for(int i=1;i<l;i++){
if(tmp=havecheck(trie[now].child,s[x][i])){
now=tmp;
}
else{
Node a;
trie.push_back(a);
sssss++;
(trie[now].child).push_back(ss(s[x][i],sssss));
now=sssss;
}
}
trie[now].yes=true;
}
void dfs(int now,int tot){
if(trie[now].yes)tot++;
if((trie[now].child).empty()){
if(tot>maxn)maxn=tot;
return;
}
for(vector<ss>::iterator it=(trie[now].child).begin();it!=(trie[now].child).end();it++)
dfs((*it).nxt,tot);
if(tot>maxn)maxn=tot;
}
void dfs2(int now){
if((trie[now].child).empty())return;
for(vector<ss>::iterator it=(trie[now].child).begin();it!=(trie[now].child).end();it++){
dfs2((*it).nxt);
}
}
int main(){
cin>>n;
for(int i=1;i<=n;i++){
cin>>s[++cnt];
if(s[cnt]==s[cnt-1])cnt--;
}
Node a;
trie.push_back(a);
for(int i=1;i<=cnt;i++){
make_tree(i);
}
dfs(0,0);
dfs2(0);
cout<<maxn<<endl;
return 0;
}
这个方法是我看题解才知道的。这个题目原来是一道栈的简单题……因为可以发现这个单词接龙与我们以往的单词接龙不同,它要求接在后面的单词一定要是前面的单词的前缀,所以我们可以先将单词按照字典序排一个序,这样,能够接上的几个单词就都在一块了。先将第一个单词入栈,再判断第二个单词能否接上。若能接上,则继续入栈。依此类推。直到有一个单词接不上前面的单词,就更新maxn为当前栈的大小,然后退栈,看这个接不上的单词与新的栈顶元素能否接上,如果不能接上,则继续退栈,直到能够接上为止,把这个单词压入栈(如果都接不上它,就让栈中只留它一个)。接下来就都一样了。因为这个方法比较简单,所以我就不附代码了。看来,学东西不能学死板了。有时候,高级算法解决不了的问题,普通算法可以轻松解决呢。