题目传送门

当时一看到这题,蒟蒻的我还以为是DP,结果发现标签是搜索……

这道题的难点在于思路和预处理,真正的搜索实现起来并不难。我们可以用一个贪心的思路,开一个dic数组记录每个单词的最小重复部分,这样搜索的时候就可以很方便地查阅dic数组,而不是每次再计算一遍。
预处理是长这样子的:

  1. void f(string a,string b,int x,int y)
  2. {
  3. int a1=a.size()-1,b1=b.size()-1;
  4. for(int i=0;i<=b1;i++) //从第一个开始枚举
  5. {
  6. if(a[0]==b[i]) //如果a的首字母和b中间的字母相同 ,则判断它们能不能接在一起
  7. {
  8. int pos=0,tot=0; //pos是当前a的第几个字母,tot是a和b的重合部分长度
  9. for(int j=i;j<=b1;j++)
  10. {
  11. if(a[pos]==b[j])
  12. {
  13. tot++;
  14. pos++;
  15. if(j==b1&&tot!=min(a1,b1)+1) //如果枚举到了最后,并且a和b没有包含关系,说明可以这么接
  16. dic[x][y]=tot; //记录最小重叠部分的长度
  17. //之所以不break,是因为后面可能还会枚举到更小的接法
  18. //比如 chsese 和 sesettt 显然 chsesesettt 要比chsesettt更优
  19. }
  20. else break;
  21. }
  22. }
  23. }
  24. }

这样就把每个单词的相互重叠部分全记录下来了,最后的处理出来的dic[x][y]是把x接在y后面的重复部分长度

之后我们就可以愉快的搜索了,搜索本身并不难,只需要注意每个单词可以用两次,以及接上新单词的“龙”的长度就可以了。

完整代码:

  1. #include<iostream>
  2. using namespace std;
  3. int n,dic[21][21],vis[21],ans;
  4. string words[21];
  5. char s;
  6. void f(string a,string b,int x,int y)
  7. {
  8. int a1=a.size()-1,b1=b.size()-1;
  9. for(int i=0;i<=b1;i++) //从第一个开始枚举
  10. {
  11. if(a[0]==b[i]) //如果a的首字母和b中间的字母相同 ,则判断它们能不能接在一起
  12. {
  13. int pos=0,tot=0; //pos是当前a的第几个字母,tot是a和b的重合部分长度
  14. for(int j=i;j<=b1;j++)
  15. {
  16. if(a[pos]==b[j])
  17. {
  18. tot++;
  19. pos++;
  20. if(j==b1&&tot!=min(a1,b1)+1) //如果枚举到了最后,并且a和b没有包含关系,说明可以这么接
  21. dic[x][y]=tot; //记录最小重叠部分的长度
  22. //之所以不break,是因为后面可能还会枚举到更小的接法
  23. //比如 chsese 和 sesettt 显然 chsesesettt 要比chsesettt更优
  24. }
  25. else break;
  26. }
  27. }
  28. }
  29. }
  30. void dfs(int pos,int sum)
  31. {
  32. ans=max(ans,sum); //实时更新ans
  33. for(int i=1;i<=n;i++)
  34. {
  35. if(dic[i][pos]&&vis[i])
  36. {
  37. vis[i]--;
  38. int suml=sum+words[i].size()-dic[i][pos]; //接上新单词"龙"的长度为=旧的长度+新单词长度-重复部分长度
  39. dfs(i,suml); //接上新单词继续搜索
  40. vis[i]++;
  41. }
  42. }
  43. }
  44. int main()
  45. {
  46. cin>>n;
  47. for(int i=1;i<=n;i++)
  48. {
  49. cin>>words[i];
  50. vis[i]=2; //初始化vis数组,每个单词能用两次
  51. }
  52. cin>>s;
  53. for(int i=1;i<=n;i++)
  54. for(int j=1;j<=n;j++)
  55. f(words[i],words[j],i,j); //预处理dic数组
  56. for(int i=1;i<=n;i++)
  57. {
  58. if(words[i][0]==s) //找到开始部分
  59. {
  60. vis[i]--;
  61. dfs(i,words[i].size()); //深搜
  62. vis[i]++;
  63. }
  64. }
  65. cout<<ans;
  66. return 0;
  67. }
  • 广告时间

在下的洛谷博客博客园博客

洛谷 P1019 单词接龙 (DFS)的更多相关文章

  1. 洛谷 P1019 单词接龙【经典DFS,温习搜索】

    P1019 单词接龙 题目描述 单词接龙是一个与我们经常玩的成语接龙相类似的游戏,现在我们已知一组单词,且给定一个开头的字母,要求出以这个字母开头的最长的“龙”(每个单词都最多在“龙”中出现两次),在 ...

  2. 洛谷 P1019 单词接龙 Label:dfs

    题目描述 单词接龙是一个与我们经常玩的成语接龙相类似的游戏,现在我们已知一组单词,且给定一个开头的字母,要求出以这个字母开头的最长的“龙”(每个单词都最多在“龙”中出现两次),在两个单词相连时,其重合 ...

  3. 洛谷P1019——单词接龙(DFS暴力搜索)

    https://www.luogu.org/problem/show?pid=1019#sub 题目描述 单词接龙是一个与我们经常玩的成语接龙相类似的游戏,现在我们已知一组单词,且给定一个开头的字母, ...

  4. 洛谷 p1019 单词接龙

    题目描述 单词接龙是一个与我们经常玩的成语接龙相类似的游戏,现在我们已知一组单词,且给定一个开头的字母,要求出以这个字母开头的最长的“龙”(每个单词都最多在“龙”中出现两次),在两个单词相连时,其重合 ...

  5. [NOIP2000] 提高组 洛谷P1019 单词接龙

    题目描述 单词接龙是一个与我们经常玩的成语接龙相类似的游戏,现在我们已知一组单词,且给定一个开头的字母,要求出以这个字母开头的最长的“龙”(每个单词都最多在“龙”中出现两次),在两个单词相连时,其重合 ...

  6. 洛谷——P1019 单词接龙(NOIP2000 T3)

    https://www.luogu.org/problem/show?pid=1019#sub 题目描述 单词接龙是一个与我们经常玩的成语接龙相类似的游戏,现在我们已知一组单词,且给定一个开头的字母, ...

  7. 洛谷P1019 单词接龙题解(超详细注释)

    https://www.luogu.org/problem/P1019 #include<cstdio> #include<cstring> #include<iostr ...

  8. 洛谷 P1019单词接龙

    看吧,多简单啊,没有人受伤的世界完成了.                                                                                ...

  9. 洛谷1019 单词接龙 字符串dfs

    问题描述 单词接龙是一个与我们经常玩的成语接龙相类似的游戏,现在我们已知一组单词,且给定一个开头的字母,要求出以这个字母开头的最长的“龙”(每个单词都最多在“龙”中出现两次),在两个单词相连时,其重合 ...

随机推荐

  1. node中的url模块解析结果

    1. URL模块作用: url 模块用于处理与解析 URL. 使用方法如下: const url = require('url'); 2. URL 字符串与 URL 对象 URL 字符串是结构化的字符 ...

  2. uoj#275. 【清华集训2016】组合数问题(数位dp)

    传送门 假设有\(k|{n\choose m}\),因为\(n!\)中质因子\(k\)的次数为\(S(n)=\left\lfloor\frac{n}{k}\right\rfloor+\left\lfl ...

  3. 密码破解工具John the Ripper使用说明

    John the Ripper John 包描述 John the Ripper 既功能丰富又运行快速. 它在一个程序中结合了几种破解模式,并且可以根据您的特定需求进行全面地配置(你甚至可以使用支持C ...

  4. C 语言实例 - 字符串翻转

    C 语言实例 - 字符串翻转 C 语言实例 C 语言实例 使用递归来翻转字符串. 实例 - 字符串翻转 #include <stdio.h> void reverseSentence(); ...

  5. Java相关书籍阅读

  6. excel输入值非法,限定了可以输入的数值怎么办

    回到excel的编辑界面,点击工具栏的“数据”标签,如图所示. 继续在“数据”标签的下面找到“数据验证”或“数据有效性”的按钮,点击该选项,然后继续下一步. 在弹出的选择框中选择“数据验证”选项,如图 ...

  7. Serervlet | 慕课课程实战 | 编写登录逻辑

    Users.java package com.po; public class Users { private String username; private String password; pu ...

  8. .NET 基础 一步步 一幕幕 [.NET基础知识点]

    .NET基础知识点   l  .Net平台  .Net FrameWork框架   l  .Net FrameWork框架提供了一个稳定的运行环境,:来保障我们.Net平台正常的运转   l  两种交 ...

  9. bryce1010专题训练——CDQ分治

    Bryce1010模板 CDQ分治 1.与普通分治的区别 普通分治中,每一个子问题只解决它本身(可以说是封闭的) 分治中,对于划分出来的两个子问题,前一个子问题用来解决后一个子问题而不是它本身 2.试 ...

  10. jQuery scrollLeft()与scrollTop() 源码解读

    这里的实现也很容易懂,通过jQuery的静态方法each给jQuery的原型添加scrollLeft和scrollTop方法. 这里在取值时它把window和普通的element做了区分 如果是win ...