洛谷 P1019 单词接龙

2024-01-07 09:52
文章标签 单词 洛谷 接龙 p1019

本文主要是介绍洛谷 P1019 单词接龙,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

题目背景

注意:本题为上古 NOIP 原题,不保证存在靠谱的做法能通过该数据范围下的所有数据。

NOIP2000 提高组 T3

题目描述

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

输入格式

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

输出格式

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

输入输出样例

输入 #1

5
at
touch
cheat
choose
tact
a

输出 #1

23

说明/提示

样例解释:连成的“龙”为 atoucheatactactouchoose

n≤20。

本题调了3个多小时才写出来,

有以下几个容易出错的地方

1,每个单词可以使用两次,

2,如果前面的龙包含后面要接的单词,这个单词不用接(接了也不增加总长度如qcr和cr)

3,单词可以不用全部用完

4,洛谷不知道什么机制不能使用 getchar();(输入开始的字符时用%s)

解题思路

本题方法深搜,搜索每一个单词,把搜索出来单词组合起来放到一个字符串数组s里面,用一个book数组标记每个单词使用的次数,按照贪心的思想,每次从s的末尾开始找是否有匹配的重合部分(保证接龙后的单词最长),dfs结束的条件是没有单词能够继续接龙下去了

代码如下

#include<stdio.h>
#include<string.h>
//top指向s的顶部,f指向v的顶部,v用来记录每次接龙除去重合部分单词的长度
int n, book[25], top = 0, v[10000], f = 0, sum = -1e9;
//a储存所有单词,b
char a[25][50], b[2], s[10000];//s为单词接龙后的组合
void dfs(int y)//直接开搜y没有特殊含义
{int i;for (i = 1; i <= n; i++)//遍历每个单词{if (book[i] < 2)//每个单词可以使用两次{int flag = 0, j;//flag标记单词是否可以接龙int k = strlen(a[i]);for (j = 0; j < k - 1; j++)//j<k-1直接排除包含情况,贪心{int h = top - 1 - j, q;if (h < 0)//s不能为空break;for (q = 0; q <= j; q++){if (a[i][q] != s[h])//如果没有重合j个字符,不能接龙break;h++;}if (h == top)//如果能够接龙{flag = 1;//标记v[f++] = k - j - 1;//求出没有重合的部分并储存break;}}if (flag == 1)//能够接龙{book[i]++;for (int u = j + 1; u < k; u++)s[top++] = a[i][u];dfs(y);//回溯一波book[i]--;top -= v[--f];}}}if (i == n + 1)//如果没有单词能够接龙,直接返回{if (top > sum){sum = top;//更新最大长度}return;}return;
}
int main()
{scanf("%d", &n);for (int i = 1; i <= n; i++)scanf("%s", a[i]);scanf("%s", &b);s[top++] = b[0];dfs(1);//dfs搜索printf("%d", sum);return 0;
}

这篇关于洛谷 P1019 单词接龙的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/579530

相关文章

每日一练7:简写单词(含链接)

1.链接 简写单词_牛客题霸_牛客网 2.题目 3.代码1(错误经验) #include <iostream>#include <string>using namespace std;int main() {string s;string ret;int count = 0;while(cin >> s)for(auto a : s){if(count == 0){if( a <=

高精度计算(代码加解析,洛谷p1601,p1303)除法待更新

目录 高精度加法 高精度减法 高精度乘法 高精度加法 我们知道在c++语言中任何数据类型都有一定的表示范围。当两个被加数很大时,正常加法不能得到精确解。在小学,我们做加法都采用竖式方法。那么我们也只需要按照加法进位的方式就能得到最终解。 8 5 6+ 2 5 5-------1 1 1 1 加法进位: c[i] = a[i] + b[i];if(c[i] >=

洛谷 凸多边形划分

T282062 凸多边形的划分 - 洛谷 | 计算机科学教育新生态 (luogu.com.cn) 先整一个半成品,高精度过两天复习一下补上 #include <iostream>#include <algorithm>#include <set>#include <cstring>#include <string>#include <vector>#include <map>

能量项链,洛谷

解释:  环形dp问题还是考虑将环拉直,可以参考我上一篇文章:环形石子合并 [2 3 5 10 2] 3 5 10 将环拉直,[]内是一个有效的区间,可以模拟吸收珠子的过程,         如[2 3 5] <=>(2,3)(3,5)    2是头,3是中间,5是尾 len >= 3:因为最后[2 10 2]是最小的可以合并的有效区间 len <= n + 1因为[2 3

动态规划---单词拆分

题目: 给你一个字符串 s 和一个字符串列表 wordDict 作为字典。如果可以利用字典中出现的一个或多个单词拼接出 s 则返回 true。 注意:不要求字典中出现的单词全部都使用,并且字典中的单词可以重复使用。 思路:本题属于完全背包问题,字符串s的长度为背包容量,字符串列表wordDict中的每一个元素相当于物品。 动态规划五部曲: 1.确定dp数组及含义 dp数组为元素类型是布

WPS如何查看已添加到词典的单词

WPS Office(12.1.0.17827) ① 点击文件,在文件中找到选项 ② 找到拼写检查并点击自定义词典 ③ 如果要删除已添加到词典的"错词",则点击修改 ④ 选择"错词", 点击删除

【C】单词长度

课程:程序设计入门——C语言(翁恺) 题目内容: 你的程序要读入一行文本,其中以空格分隔为若干个单词,以‘.’结束。你要输出这行文本中每个单词的长度。这里的单词与语言无关,可以包括各种符号,比如“it's”算一个单词,长度为4。注意,行中可能出现连续的空格。 输入格式: 输入在一行中给出一行文本,以‘.’结束,结尾的句号不能计算在最后一个单词的长度内。 输出格式: 在一行中输出这行

代码随想录第八天|151.翻转字符串里的单词 卡码网:55.右旋转字符串 28. 实现 strStr() 459.重复的子字符串

反转字符串的单词 思路:刷过稍微忘记 class Solution {public://去除空格string remove(string s){//使用快慢指针int slow=0;int i=0;for(;i<s.size();i++){if(s[i]!=' '){if(slow!=0){s[slow++]=' ';}while(s[i]!=' '&&i<s.size()){s[slow+

洛谷P5490扫描线

0是最小的数字,将一个线段看成一个区间,对于一个矩形,从下扫到上,入边为1,而出边为-1,意思是将这个区间上的所有点加1(区间修改).把线段表示为Line[i],其中记录了l,r,h,tag,左右端点,高度,入边还是出边(1或-1) 那么每次区间修改后不为0的区间它的值可能是1,2,3或者是其它数字,这不好统计,可以将它转化一下,0是不是表示没有被覆盖过的地方,我们只要统计0的个数然后用总长减去

实操在聆思CSK6大模型开发板的英文评测SDK中自定义添加单词、短语、句子资源

引言 英文评测示例通过对用户语音输入的英文单词进行精准识别,提供 单词、短语、句子 三种类型,用户在选择好类型后,可根据屏幕上的提示进行语音输入,评测算法将对输入的英文语音进行精准识别,并对单词的发音、错读、漏读、多读等方面进行评估。 本文将详细介绍在聆思CSK6大模型语音视觉开发板上,如何替换英文评测示例中的单词、短语和句子,从而让您有更好的AI应用体验。 ·· 获取英文评测SDK 部