洛谷|洛谷 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
推荐阅读
- 背单词是噩梦(不着急,比“红宝书”更好的记忆单词的方法来了!——从词源的角度记单词(二))
- 墨墨单词/2018.3.6/
- SCI论文写作怎样巧用英语单词--Editideas(辑思编译)
- 汤世声(学好英语,从提高记忆力开始)
- 语音篇?重读位置(七)
- 2018.05.08
- Android命名规范
- 72.编辑距离
- 曲根的北美英语单词|曲根的北美英语单词 6 7
- LeetCode-139-单词拆分