P1278 单词游戏 简单搜索+玄学优化

单词游戏

传送门

题目描述

Io 和 Ao 在玩一个单词游戏。

他们轮流说出一个仅包含元音字母的单词,并且后一个单词的第一个字母必须与前一个单词的最后一个字母一致。

游戏可以从任何一个单词开始。

任何单词禁止说两遍,游戏中只能使用给定词典中含有的单词。

游戏的复杂度定义为游戏中所使用的单词长度总和。

编写程序,求出使用一本给定的词典来玩这个游戏所能达到的游戏最大可能复杂度。

输入格式

输入文件的第一行,表示一个自然数 N ( 1 ≤ N ≤ 16 ) N(1 \le N \le 16) N(1≤N≤16), N N N 表示一本字典中包含的单词数量以下的每一行包含字典中的一个单词,每一个单词是由字母 AEIOU 组成的一个字符串,每个单词的长度将小于等于 100 100 100,所有的单词是不一样的。

输出格式

输出文件仅有一行,表示该游戏的最大可能复杂度。

样例 #1

样例输入 #1

复制代码
5
IOO
IUUO
AI
OIOOI
AOOI

样例输出 #1

复制代码
16

注明

以上来自洛谷。 以上来自洛谷。 以上来自洛谷。

解题思路

直接找题意做即可。循环遍历每一个字符串,将其设为第一个字符串,然后暴搜就行了。注意当搜索次数达到 5 × 1 0 7 5\times 10^7 5×107 时,直接输出答案并结束程序。说起来有点抽象,结合代码容易理解。

AC Code

cpp 复制代码
#include<bits/stdc++.h>
using namespace std;
int n;
string s[20];
int _s[20];
bool vis[20];
int COUNT;
int Answer;
inline void dfs(int len, char c) {
	++COUNT;
	if (COUNT >= 5e7)cout << Answer, exit(0);
	Answer = max(Answer, len);
	for (register int i = 1; i <= n; ++i) {
		if (vis[i])continue;
		if (s[i][0] == c) {
			vis[i] = 1;
			dfs(len + _s[i], s[i][_s[i] - 1]);
			vis[i] = 0;
		}
	}
}
signed main() {
	cin >> n;
	for (register int i = 1; i <= n; ++i)cin >> s[i], _s[i] = s[i].size();
	for (register int i = 1; i <= n; ++i) {
		vis[i] = 1;
		dfs(_s[i], s[i][_s[i] - 1]);
		vis[i] = 0;
	}
	cout << Answer << endl;
	return 0;
}

闲话

一开始以为这题非常简单,没想到这么简单。注意到洛谷难度评级为:

#@!&( #@&(@!$(!@&#()&*%%#&Y&&%%%#%@!&(<-开启词汇联想)

大佬 C Wx 用状压 DP 通过本题,牛炸了。

相关推荐
小辉懂编程10 分钟前
C语言:51单片机实现数码管依次循环显示【1~F】课堂练习
c语言·开发语言·51单片机
醍醐三叶1 小时前
C++类与对象--2 对象的初始化和清理
开发语言·c++
Inverse1622 小时前
C语言_动态内存管理
c语言·数据结构·算法
wuqingshun3141593 小时前
蓝桥杯 16. 外卖店优先级
c++·算法·职场和发展·蓝桥杯·深度优先
海绵宝宝贾克斯儿3 小时前
C++中如何实现一个单例模式?
开发语言·c++·单例模式
Epiphany.5563 小时前
素数筛(欧拉筛算法)
c++·算法·图论
龙湾开发3 小时前
计算机图形学编程(使用OpenGL和C++)(第2版)学习笔记 10.增强表面细节(二)法线贴图
c++·笔记·学习·图形渲染·贴图
whoarethenext4 小时前
c/c++的opencv的轮廓匹配初识
c语言·c++·opencv
爱吃涮毛肚的肥肥(暂时吃不了版)4 小时前
项目班——0510——JSON网络封装
c++·算法·json
apocelipes4 小时前
使用libdivide加速整数除法运算
c语言·c++·性能优化·linux编程