题目描述
单词接龙是一个与我们经常玩的成语接龙相类似的游戏,现在我们已知一组单词,且给定一个开头的字母,要求出以这个字母开头的最长的“龙”(每个单词都最多在“龙”中出现两次),在两个单词相连时,其重合部分合为一部分,例如 beast和tonish,如果接成一条龙则变为beastonish,另外相邻的两部分不能存在包含关系,例如at 和 atid 间不能相连。
输入格式
输入的第一行为一个单独的整数n (n≤20)表示单词数,以下n 行每行有一个单词,输入的最后一行为一个单个字符,表示“龙”开头的字母。你可以假定以此字母开头的“龙”一定存在.
输出格式
只需输出以此字母开头的最长的“龙”的长度
输入输出样例
输入 #1
5
at
touch
cheat
choose
tact
a
输出 #1
23
说明/提示
(连成的“龙”为atoucheatactactouchoose)
第一眼就感觉是dfs(以后试试欧拉图能不能写)
写了之后发现这题其实是string方法的综合应用。。。(手动黑脸)
这里写上所用的string方法的用法:
string::substr(int start=0,int lenth=nops) 会返回一个从原字符串下标为start开始长度为len的一个子串(默认起始位置为0,默认长度为(unsigned)-1)
string::erase(int start,int len) 将原字符串从下标为start的位置开始删除len位
知道了这些剩下的就是dfs模拟了。。。
ACCODE
#include <iostream>
using namespace std;
string k[50];
int vis[50]={0};
int n;
int serch(string a,string b)
{
for(int i=1;i<=min(a.size(),b.size());i++)
{
// cout<<b.substr(0,i)<<" "<<a.substr(a.size()-i)<<endl;
if(b.substr(0,i)==a.substr(a.size()-i))
{
return i;
}
}
}
int ans=0;
void dfs(string str)
{
for(int i=0;i<2*n;i++)
{
int len;
string t=str;
// cout<<t<<"&"<<k[i]<<endl;
if(vis[i]) continue;
else if(len=serch(t,k[i]))
{
// cout<<len;
vis[i]=1;
t.erase(t.size()-len,len);
t+=k[i];
// cout<<t<<endl;
dfs(t);
vis[i]=0;
}
}
if(ans<str.size())
{
ans=str.size();
}
}
int main()
{
cin>>n;
for(int i=0;i<n;i++)
{
cin>>k[i];
k[i+n]=k[i];
}
string str;
cin>>str;
dfs(str);
cout<<ans;
return 0;
}
坑点
- 有多种合并情况的取合并后最长的字符串 栗:ababab 与 ababc 应合并为 ababababc 而不是abababc。。。
- string::size()方法返回一个无符号数 。Int 类型的数会转换为无符号数再和string::size()比较,所以ans初始化为-1的话。。