洛谷P1019单词接龙(dfs+string应用)

题目描述

单词接龙是一个与我们经常玩的成语接龙相类似的游戏,现在我们已知一组单词,且给定一个开头的字母,要求出以这个字母开头的最长的“龙”(每个单词都最多在“龙”中出现两次),在两个单词相连时,其重合部分合为一部分,例如 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的话。。
发布了30 篇原创文章 · 获赞 9 · 访问量 1321

猜你喜欢

转载自blog.csdn.net/Zhang_sir00/article/details/99745016