洛谷千题详解 | P1019 [NOIP2000 提高组] 单词接龙【C++、Java语言】

博主主页:Yu·仙笙

专栏地址:洛谷千题详解

目录

题目描述

输入格式

输出格式

输入输出样例

 解析:

C++源码:

Java源码:


--------------------------------------------------------------------------------------------------------------------------------

 --------------------------------------------------------------------------------------------------------------------------------

题目描述

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

 --------------------------------------------------------------------------------------------------------------------------------

输入格式

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

 --------------------------------------------------------------------------------------------------------------------------------

输出格式

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

 --------------------------------------------------------------------------------------------------------------------------------

输入输出样例

输入 #1

5
at
touch
cheat
choose
tact
a

输出 #1

 --------------------------------------------------------------------------------------------------------------------------------

 解析:

谨以此题解纪念我的洛谷橙名,我会继续努力。

基本思路是搜索。

处理的难点在于对重叠部分的处理。

单词的使用次数很好判断,开一个数组即可,和正常向dfs的vis数组差不多。

但对于重叠部分的处理,我想细说一下。

因为连接起来的单词要最长,所以对比是选择从上一个单词的末尾与当前单词的开头进行比对,如果发现不符那就不能匹配。

这里我借鉴了一位大神的思路,使用一个check函数,用来比较两个串s和m的长度为k的接口能不能匹配。

判断方式有两种,第一种就是我用的这样按字符比较,如果发现某处不匹配立即返回false。还有一个方法是用string里面的substr,我就不展开了,有兴趣的同学可以试一下。

我们假设接口长度为k,串s的长度为lens,然后我们从0到k枚举,判断s[lens-k+i]是不是等于m[i]。

这个式子怎么来的呢?接口前面的串是s,后面的串是m,那么很显然,s串的接口最开始应该是lens-k处,然后在后面加上一个枚举的i就可以保证扫描到所有接口字符(我们的i是从0开始枚举的)。

还有一些细节问题。 如果我们在接龙的时候发现我们现在要接的龙还不如之前某一次接过的长,那么这个接龙方案肯定不是最优的,所以要舍去,这个正确性是显然的。

(想一下深搜时的遍历过程,这句话便不难理解)

使用我这个方法,可能有一个奇怪的想法有同学没想到,那就是在进行拼接操作时,要注意使用给定的串的副本(即复制一份原来的串)进行拼接处理。

这是因为,如果你把原串改变了,而且这个串还不是最优的,那就完了,回溯不回去了。

具体到操作上,首先,我们从1到n枚举每个短字符串,如果它已经被用了两次则continue,然后我们求出当前短串的长度,从1到这个长度枚举,枚举的是接口的长度(自然是接口越短融合串越长嘛) 然后执行拼接操作,记录一下最大长度,再加上回溯就好了。

拼接操作和check函数很像,具体到代码上大家就能看明白了。

 --------------------------------------------------------------------------------------------------------------------------------

C++源码:

#include<cstdio>
#include<iostream>
#include<string>
#include<cmath> 
using namespace std;
int n;//单词数 
string tr[30];//存储字符串 
int yc[30][30];//两个字母的最小重叠部分 
int vis[30];//判断单词使用频率. 
int mt(int x, int y){//mt函数,返回x单词后连接一个y单词的最小重叠部分 
    bool pp=true; 
    int ky=0;
    for(int k=tr[x].size()-1;k>=0;k--){//从x单词尾部向前看看最小重叠部分是从哪里开始的,以为因为是倒着来,所以保证是最小的 
        for(int kx=k;kx<tr[x].size();kx++){/ 
            if(tr[x][kx]!=tr[y][ky++]){
                pp=false;
                break;
            }
        }
        if(pp==true){//如果说当前以k为开头的前一个单词后缀 ,是后面单词的前缀,就马上返回重叠部分。(tr[x].size()-k是找出来的规律)
            return tr[x].size()-k;        } 
        ky=0;
        pp=true;//不行就继续
    }
    return 0;
}//可能这里有点难理解。可以手动模拟一下
char ch;//开头字母 
int ans=-1;//答案 
int an=0;//每次搜到的当前最长串 
void dfs(int p){//p为尾部单词编号(p的后缀就是“龙”的后缀,因为p已经连接到”龙“后面了)
    bool jx=false; 
    for(int j=1;j<=n;j++){
        if(vis[j]>=2) continue;//使用了两次就跳过 
        if(yc[p][j]==0) continue;//两单词之间没有重合部分就跳过 
        if(yc[p][j]==tr[p].size() || yc[p][j]==tr[j].size()) continue;//两者存在包含关系就跳过 
        an+=tr[j].size()-yc[p][j];//两单词合并再减去最小重合部分 
        vis[j]++;//使用了一次
        jx=true;//标记一下当前已经成功匹配到一个可以连接的部分 
        dfs(j); //接上去
        an-=tr[j].size()-yc[p][j];//回溯,就要再减回去那一部分长度 
        vis[j]--;//回溯,使用-- 
    }
    if(jx==false){//jx==false说明不能再找到任何一个单词可以相连了 
        ans=max(ans,an);//更新ans 
    }
    return;
}
int main(){
    scanf("%d",&n);
    for(int i=1;i<=n;i++)
        cin>>tr[i];
    cin>>ch; 
    for(int i=1;i<=n;i++){
        for(int j=1;j<=n;j++){
            yc[i][j]=mt(i,j); 
        }
    }//预处理yc数组。yc[i][j]就表示,i单词后连接一个j单词的最小重叠部分 
    //比如 i表示at,j表示att. yc[i][j]就为2 但是yc[j][i]就为0.
    //预处理是一个关键
     
    for(int i=1;i<=n;i++){//从头到尾看一下有没有以指定开头字母为开头的单词 
        if(tr[i][0]==ch){//如果有,就以当前单词为基准进行搜索。 
            vis[i]++;//使用过一次 
            an=tr[i].size();//更新当前串长度 
            dfs(i);//接上
            vis[i]=0;//消除影响 
        } 
    } 
    printf("%d",ans);
    return 0;
}
#include <iostream>
#include <cstdio>
#include <cstring>
#include <string>
#define maxn 100
using namespace std;
int n;
int ans = 0;
string word[maxn];//字符串数组,用来存储单词
string beginn;//用来存储开头字符
int used[maxn];//这个就是用来记录dfs时候每一个单词被使用了几次的数组

bool check(string s,string m,int k){//重点一,check函数判断接口可行性,k代表接口长度,以下同
    int lens = s.length();
    for (int i=0;i<k;i++){
        if(s[lens-k+i]!=m[i])//我讲过了
            return false;
    }
    return true;
}

void add(string &s,string m,int k){//拼接操作,因为要把m接到s上,所以对于s串不可以传参,因为我们要试图改变这个串
    int lenm = m.length();
    for (int i=k;i<lenm;i++)
        s+=m[i];//C++字符串类型特性操作,支持+=符号直接拼接
}

void dfs(string now){//这只是一个看似平淡无奇的dfs
    int x = now.length();
    ans = max(ans,x);//每次拼接之后更新一下答案
    for (int i=1;i<=n;i++){
        if (used[i]>=2)//如果有一个单词用完了,那这个单词就不能选了
            continue;
        int maxk = word[i].length();
        for (int j=1;j<=maxk;j++){//枚举接口长度
            if (check(now,word[i],j)){
                string temp = now;//重点二,使用字符串副本进行拼接
                add(temp,word[i],j);
                if (temp==now)//拼完之后如果发现长度没增加,也就是和原串一样,那这次拼接没有意义,剪掉
                    continue;
                used[i]++;
                dfs(temp);
                used[i]--;//这只是一个看似平淡无奇的回溯
            }
        }
    }
    
}

int main(){
    cin >> n;
    for (int i=1;i<=n;i++)
        cin >> word[i];
    cin >> beginn;
    dfs(beginn);
    
    cout << ans << endl;
    return 0;
    
}

 --------------------------------------------------------------------------------------------------------------------------------

Java源码:

import java.util.Scanner;

public class P1019 {
// 要点:
//	1.单词拼接一定要相同的部分:能连接
//	2.在保证能连接的情况下,保证连接长度最长
//	3.不能存在包含关系
	static String []a=new String[30];//存储单词
	static int []use_times=new int[30];//使用次数
	static int p=0;
	static int n=0,ans=0;//N为单词数,ans是长度
	
	//用于查找两字符串是否是包含关系,或者相同的有多少个单词
	public static int  find(int index,int y) {//y为当前匹配的单词,index为龙当前的最后一个单词
		
		for(int i=a[index].length()-1;i>=1;i--) {//查找龙的最后一个单词从尾部向前搜索和匹配单词的开头相同的位置

			if(a[index].substring(i, i+1).equals(a[y].substring(0,1))) {
				
				int k=0;//设立临时值,用于单词匹配数
				for(int j=i+1;j<a[index].length();j++) {//如果龙的最后一个匹配,则直接退出,保证单词连接最长。
					k++;
					if(!a[index].substring(j,j+1).equals(a[y].substring(k,k+1))) {
						//如果某个字母不相同,表示两个字符串没有公共部分。无法连接在一起
						return 0;
					}
					
				}
				return a[y].length()-k-1;//能到这,最起码都是第一个匹配成功
			}
		}
		return 0;
	}
	public static void dfs(int x) {//x为当前位置
		for(int i=1;i<=n;i++) {
			if(use_times[i]<2&&find(x,i)>0) {//单词不能用两次,并且不能包含:表示连接
				use_times[i]++;//使用次数加一
				ans+=find(x,i);//加上新单词长度
				dfs(i);//接着这个单词搜索
				use_times[i]--;//回溯
				ans-=find(x,i);
				
			}
		}
		if(ans>p)
			p=ans;
		
	}
	  
	  
	
 public static void main(String[] args)  {
  Scanner cin =new Scanner(System.in);
   n=cin.nextInt();//单词数
   int maxn=0;
  for(int i=1;i<=n;i++) {//存储单词
	  a[i]=cin.next();
  }
  String b=cin.next();//龙的开头
  for(int i = 1; i <= n; i++)
  {
      if(a[i].substring(0,1).equals(b))//查找首字母为龙的开头的单词
      {	
          ans = a[i].length();//先加上首单词的长度 
          use_times[i]++;//首单词用过了 
          dfs(i);//以第i个单词为开头,尝试计算长度
          use_times[i]--;//回溯 
          if(p > maxn)//求最大 
          {
              maxn = p;
          }
      }
  }
  System.out.println(maxn);
  cin.close();
	  
 }
}

 --------------------------------------------------------------------------------------------------------------------------------

题目描述 给出一单词,按照以下规则进行单词接龙: 1.首先任意选定一个单词作为开头; 2.接下来每次接的单词必须满足:它的首字母与上一个单词的尾字母相同; 3.同一个单词不能重复使用; 4.当无法继续接下去时,游戏结束。 你的任务是编写一个程序,计算出在给定单词集合中,最长的单词接龙长度。 输入格式 输入文件的第一行为一个整数n,表示单词的个数。 接下来n行,每行一个单词,由小写字母成,长度不超过20个字符。 输出格式 输出文件仅一行,包含一个整数,表示最长的单词接龙长度。 数据范围 1≤n≤500 输入样例 5 ab bc cd de ef 输出样例 5 提示 输入样例2: 6 a b c d e f 输出样例2: 1 输入样例3: 4 aa bb cc dd 输出样例3: 2 输入样例4: 4 a b ba ab 输出样例4: 2 输入样例5: 3 a aa aaa 输出样例5: 1 输入样例6: 3 a aa ab 输出样例6: 2 输入样例7: 4 a aa aaa aaaa 输出样例7: 1 输入样例8: 4 a aa aaa baaa 输出样例8: 2 输入样例9: 4 a aa baa ab 输出样例9: 2 输入样例10: 4 a aa baa ba 输出样例10: 2 输入样例11: 4 a aa baa bb 输出样例11: 2 输入样例12: 4 a aa baa bc 输出样例12: 2 输入样例13: 4 a aa baa bca 输出样例13: 2 输入样例14: 4 a aa baa bcb 输出样例14: 2 输入样例15: 4 a aa baa bcc 输出样例15: 2 输入样例16: 4 a aa baa bcdc 输出样例16: 2 输入样例17: 4 a aa baa bcde 输出样例17: 2 输入样例18: 4 a aa baa bcdef 输出样例18: 2 输入样例19: 4 a aa baa bcdefg 输出样例19: 2 输入样例20: 4 a aa baa bcdefgh 输出样例20: 2 输入样例21: 4 a aa baa bcdefghi 输出样例21: 2 输入样例22: 4 a aa baa bcdefghij 输出样例22: 2 输入样例23: 4 a aa baa bcdefghijk 输出样例23: 2 输入样例24: 4 a aa baa bcdefghijkl 输出样例24: 2 输入样例25: 4 a aa baa bcdefghijklm 输出样例25: 2 输入样例26: 4 a aa baa bcdefghijklmn 输出样例26: 2 输入样例27: 4 a aa baa bcdefghijklmno 输出样例27: 2 输入样例28: 4 a aa baa bcdefghijklmnop 输出样例28: 2 输入样例29: 4 a aa baa bcdefghijklmnopq 输出样例29: 2 输入样例30: 4 a aa baa bcdefghijklmnopqr 输出样例30: 2 输入样例31: 4 a aa baa bcdefghijklmnopqrs 输出样例31: 2 输入样例32: 4 a aa baa bcdefghijklmnopqrst 输出样例32: 2 输入样例33: 4 a aa baa bcdefghijklmnopqrstu 输出样例33: 2 输入样例34: 4 a aa baa bcdefghijklmnopqrstuv 输出样例34: 2 输入样例35: 4 a aa baa bcdefghijklmnopqrstuvw 输出样例35: 2 输入样例36: 4 a aa baa bcdefghijklmnopqrstuvwx 输出样例36: 2 输入样例37: 4 a aa baa bcdefghijklmnopqrstuvwxy 输出样例37: 2 输入样例38: 4 a aa baa bcdefghijklmnopqrstuvwxyz 输出样例38: 2 输入样例39: 4 a aa baa bcdefghijklmnopqrstuvwxyzA 输出样例39: 2 输入样例40: 4 a aa baa bcdefghijklmnopqrstuvwxyzAB 输出样例40: 2 输入样例41: 4 a aa baa bcdefghijklmnopqrstuvwxyzABC 输出样例41: 2 输入样例42: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCD 输出样例42: 2 输入样例43: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDE 输出样例43: 2 输入样例44: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEF 输出样例44: 2 输入样例45: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFG 输出样例45: 2 输入样例46: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGH 输出样例46: 2 输入样例47: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHI 输出样例47: 2 输入样例48: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJ 输出样例48: 2 输入样例49: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJK 输出样例49: 2 输入样例50: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKL 输出样例50: 2 输入样例51: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLM 输出样例51: 2 输入样例52: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMN 输出样例52: 2 输入样例53: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNO 输出样例53: 2 输入样例54: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOP 输出样例54: 2 输入样例55: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQ 输出样例55: 2 输入样例56: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQR 输出样例56: 2 输入样例57: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRS 输出样例57: 2 输入样例58: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRST 输出样例58: 2 输入样例59: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTU 输出样例59: 2 输入样例60: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUV 输出样例60: 2 输入样例61: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVW 输出样例61: 2 输入样例62: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWX 输出样例62: 2 输入样例63: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXY 输出样例63: 2 输入样例64: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ 输出样例64: 2 输入样例65: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZA 输出样例65: 2 输入样例66: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZAB 输出样例66: 2 输入样例67: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZABC 输出样例67: 2 输入样例68: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZABCD 输出样例68: 2 输入样例69: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZABCDE 输出样例69: 2 输入样例70: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZABCDEF 输出样例70: 2 输入样例71: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZABCDEFG 输出样例71: 2 输入样例72: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZABCDEFGH 输出样例72: 2 输入样例73: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZABCDEFGHI 输出样例73: 2 输入样例74: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZABCDEFGHIJ 输出样例74: 2 输入样例75: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZABCDEFGHIJK 输出样例75: 2 输入样例76: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZABCDEFGHIJKL 输出样例76: 2 输入样例77: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZABCDEFGHIJKLM 输出样例77: 2 输入样例78: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZABCDEFGHIJKLMN 输出样例78: 2 输入样例79: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZABCDEFGHIJKLMNO 输出样例79: 2 输入样例80: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZABCDEFGHIJKLMNOP 输出样例80: 2 输入样例81: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZABCDEFGHIJKLMNOPQ 输出样例81: 2 输入样例82: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZABCDEFGHIJKLMNOPQR 输出样例82: 2 输入样例83: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZABCDEFGHIJKLMNOPQRS 输出样例83: 2 输入样例84: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZABCDEFGHIJKLMNOPQRST 输出样例84: 2 输入样例85: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZABCDEFGHIJKLMNOPQRSTU 输出样例85: 2 输入样例86: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZABCDEFGHIJKLMNOPQRSTUV 输出样例86: 2 输入样例87: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZABCDEFGHIJKLMNOPQRSTUVW 输出样例87: 2 输入样例88: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZABCDEFGHIJKLMNOPQRSTUVWX 输出样例88: 2 输入样例89: 4 a aa baa bcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZABCDEFGHIJKLMNOPQRSTUVWXY 输出样例89: 2 输入样例90: 4
评论 30
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

.LAL.

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值