洛谷|洛谷 P1019 单词接龙 深搜

题目描述 单词接龙是一个与我们经常玩的成语接龙相类似的游戏,现在我们已知一组单词,且给定一个开头的字母,要求出以这个字母开头的最长的“龙”(每个单词都最多在“龙”中出现两次),在两个单词相连时,其重合部分合为一部分,例如 beast和astonish,如果接成一条龙则变为beastonish,另外相邻的两部分不能存在包含关系,例如at 和 atide 间不能相连。
输入输出格式 输入格式:

输入的第一行为一个单独的整数n (n<=20)表示单词数,以下n 行每行有一个单词,输入的最后一行为一个单个字符,表示“龙”开头的字母。你可以假定以此字母开头的“龙”一定存在.

输出格式:

只需输出以此字母开头的最长的“龙”的长度

输入输出样例 输入样例#1:

5 at touch cheat choose tact a

输出样例#1:
23(连成的“龙”为atoucheatactactouchoose)



思路:
将单词看成点,如果 a 单词后面能接上 b 单词,则 a 到 b 有一条边,边的长度为 a 后面和 b 前面最短的公共部分,这样能使最后的长度最长。
#include #include using namespace std; const int MAX = 25; char s[MAX][2005]; int G[MAX][MAX]; //可以将问题转化成图 int vis[MAX]; int n; char start; int ans; int getEdge(char *s1, char *s2); // s1 是否可以到 s2,若可以,返回重叠部分的字符数,若不可以,返回 0void dfs(int i, int len); int main(){ //freopen("input.txt", "r", stdin); cin >> n; for(int i=1; i<=n; i++){ cin >> s[i]; } //初始化图 for(int i=1; i<=n; i++){ for(int j=1; j<=n; j++){ G[i][j] = getEdge(s[i], s[j]); } }//深搜 cin >> start; ans = 0; for(int i=1; i<=n; i++){ if(s[i][0] == start){ memset(vis, 0, sizeof(vis)); dfs(i, strlen(s[i])); } }cout << ans; return 0; } int getEdge(char *s1, char *s2){ //从 s1 第二个字母开始,s2 不能包含在 s1 中 int len1 = strlen(s1), len2 = strlen(s2); for(int i=len1-1; i>=1; i--){//公共部分越少越好 int n1, n2; for(n1=i, n2=0; n1

【洛谷|洛谷 P1019 单词接龙 深搜】转载于:https://www.cnblogs.com/lighter-blog/p/7287185.html

    推荐阅读