7.9 单词游戏
源程序名 words.???(pas, c, cpp)
可执行文件名 words.exe
输入文件名 words.in
输出文件名 words.out
【问题描述】
Io和Ao在玩一个单词游戏。
他们轮流说出一个仅包含元音字母的单词,并且后一个单词的第一个字母必须与前一个单词的最后一个字母一致。
游戏可以从任何一个单词开始。
任何单词禁止说两遍,游戏中只能使用给定词典中含有的单词。
游戏的复杂度定义为游戏中所使用的单词长度总和。
编写程序,求出使用一本给定的词典来玩这个游戏所能达到的游戏最大可能复杂度。
【输入】
输入文件的第一行,表示一个自然数N(1≤N≤16),N表示一本字典中包含的单词数量以下的每一行包含字典中的一个单词,每一个单词是由字母A、E、I、O和U组成的一个字符串,每个单词的长度将小于等于100,所有的单词是不一样的。
【输出】
输出文件仅有一行,表示该游戏的最大可能复杂度。
【样例】
words.in words.out
5 16
IOO
IUUO
AI
OIOOI
AOOI
【问题分析】
枚举要取的单词,如果这个单词构成的图存在欧拉路(不一定是回路),则存在一个单词接龙的方案,这些单词的长度和就是一个解。答案是所有解的最大值。算法的时间复杂度是O(2n•n),n≤16。
这道题用搜索过7组,加了优化(去掉不可能的单词),不知道为什么还是7组
#include <cstdio>
char str[18][102];
bool used[18];
long n;
long max = -0x7fff0000;
bool remain[18];
void dfs(long l,long s)
{
bool flag = false;
for (long i=1;i<n+1;i++)
{
if (!used[i]&&str[l][str[l][0]]==str[i][1])
{
flag = true;
used[i] = true;
dfs(i,s+str[i][0]);
used[i] = false;
}
}
if (!flag)
{
if (s>max)
max = s;
}
}
int main()
{
freopen("words.in","r",stdin);
freopen("words.out","w",stdout);
scanf("%ld",&n);
for (long i=1;i<n+1;i++)
{
scanf("%s",str[i]+1);
}
for (long i=1;i<n+1;i++)
{
while (str[i][++str[i][0]]!=0);
str[i][0]--;
}
for (long i=1;i<n+1;i++)
{
for (long j=1;j<n+1;j++)
{
if (i==j) continue;
if (str[i][1]==str[j][str[j][0]]
||str[i][str[j][0]]==str[i][1])
{
remain[i] = remain[j] = true;
break;
}
}
}
for (long i=1;i<n+1;i++)
{
if (!remain[i]) continue;
used[i] = true;
dfs(i,str[i][0]);
used[i] = false;
}
printf("%ld",max);
//system("PAUSE");
return 0;
}