vlak
2023-7-14
题目
题目描述
Nina 和 Emilija 正在玩一个特殊的游戏。这个游戏是在一张最开始为空白的纸上进行的。在每一个人的行动回合内,这个人会在这张纸上当前的单词后面加入一个字母。她们会轮流行动,而 Nina 先手行动。
操作者必须保证这样一个条件:在添加完一个字符后,整张纸上的单词必须是操作人最喜欢的歌曲的一个单词的前缀。如果不满足条件,进行这个操作的人就输了。
你的问题是,如果两个人都采取最优策略,那么谁会获得最后的胜利。
输入格式
第一行包含一个正整数 \(n\),表示 Nina 最喜爱歌曲中的单词数量。
接下来的 \(n\) 行,每一行输入 Nina 最喜爱歌曲中的一个单词。
接下来的一行包含一个正整数 \(m\),表示 Emilija 最喜爱歌曲中的单词数量。
接下来的 \(m\) 行,每一行输入 Emilija 最喜爱歌曲中的一个单词。
输入的单词都只包含小写字母,且单词长度总和不超过 \(200000\)。
输出格式
输出获胜的玩家,Nina 或 Emilija。
样例输入
2
aaa
bbb
3
aab
aba
bbb
样例输出
Nina
难度&重要性(1~10):5.5
题目来源
COCI,luogu
题目算法
字典树
解题思路
因为它要求必须是为喜欢的歌的前缀,那么我们就很容易想到建一个字典树记录所有歌。
在记录所有歌的同时,再用一个变量来表示这一段前缀是公共的,还是只有 Nina 或 Emilija 有。
例如样例数据:
2
aaa
bbb
3
aab
aba
bbb
建出的字典树如图:

这里的 \(1\) 表示的是 Nina 的前缀,\(2\) 表示的是 Emilija 的前缀,\(3\) 表示是公共前缀。
因为是回合制的游戏,所以字典树的相邻两层是由不同的人来选择的。
我们只需要在字典树上跑一个 dfs,每一次来判断上一层的人的选择是否可行。
即如果第 \(i\) 层的第 \(j\) 个字符,如果这个人可以通过选下一层的一个字符获胜,那么就说明另一个人不应该选择第 \(i\) 层的第 \(j\) 个字符。
反之如果这个人怎么选都不可以获胜,那么就说明另一个人可以通过第 \(i\) 层的第 \(j\) 个字符获胜。
Code
void make(int k){//建字典树
int len=s.size(),u=0;
For(i,0,len-1){
int c=s[i]-'a';
if(!ch[u][c].i)ch[u][c].i=++tot;
if(ch[u][c].k==0)ch[u][c].k=k;//如何没有出现过,那就只有这一个人
if(ch[u][c].k!=k)ch[u][c].k=3;//如何另一个人也有这个前缀,那这就是公共的
u=ch[u][c].i;
}
return;
}
bool dfs(int u,int p){
int k=0;
For(i,0,25)
if(ch[u][i].i!=0){
if(ch[u][i].k==3)k=dfs(ch[u][i].i,!p);
if(ch[u][i].k==1&&p==0)return 0;
if(ch[u][i].k==2&&p==1)return 0;
if(k)return 0;
//如果p这个人可以获胜,即另一个人不应选u,返回0
}
return 1;
//如果p这个人怎么选都获胜不了,那么另一个人就可以选u,返回1
}
signed main(){
cin>>n;
For(i,1,n){
cin>>s;
make(1);//1表示 Nina
}
cin>>m;
For(i,1,m){
cin>>s;
make(2);//2表示 Emilija
}
For(i,0,25)
if(ch[0][i].i!=0){
ans=dfs(ch[0][i].i,1);
//0表示 Nina
//1表示 Emilija
if(ans){//如何可以获胜就输出 Nina
cout<<"Nina";
return 0;
}
}
//如何 Nina怎么选都不能获胜,就输出 Emilija
cout<<"Emilija";
return 0;
}
完成状态
已完成
vlak的更多相关文章
- Centos下安装LoadRunner负载机
aaarticlea/png;base64,iVBORw0KGgoAAAANSUhEUgAAAd0AAADfCAIAAAA84J6GAAAgAElEQVR4nJzb93cbR6In+uK/8v6Fd8
随机推荐
- 2021-04-18:给定一个二维数组matrix,里面的值不是1就是0,上、下、左、右相邻的1认为是一片岛,返回matrix中岛的数量。
2021-04-18:给定一个二维数组matrix,里面的值不是1就是0,上.下.左.右相邻的1认为是一片岛,返回matrix中岛的数量. 福大大 答案2021-04-18: 并查集. 代码用gola ...
- 2021-12-13:字符串解码。给定一个经过编码的字符串,返回它解码后的字符串。 编码规则为: k[encoded_string],表示其中方括号内部的 encoded_string 正好重复 k
2021-12-13:字符串解码.给定一个经过编码的字符串,返回它解码后的字符串. 编码规则为: k[encoded_string],表示其中方括号内部的 encoded_string 正好重复 k ...
- django之drf(部分讲解)
序列化类常用字段和字段参数 drf在Django字段类型的基础上派生了自己的字段类型以及字段参数 序列化器的字段类型用于处理原始值和内部数据类型直接的转换 还可以用于验证输入.以及父对象检索和设置值 ...
- c++函数参数和返回值
c++函数参数和返回值 函数存储位置 函数参数入栈顺序 初始化列表 函数的返回值 用参数引用来返回 返回一个参数指针 返回一个对象 总结 函数的几种变体 inline 函数 函数对象 lambda 函 ...
- L2-001 紧急救援 (25 分)
1.题目描述: 作为一个城市的应急救援队伍的负责人,你有一张特殊的全国地图.在地图上显示有多个分散的城市和一些连接城市的快速道路.每个城市的救援队数量和每一条连接两个城市的快速道路长度都标在地图上.当 ...
- 【LeetCode双向链表】LRU详解,双向链表实战
LRU缓存 请你设计并实现一个满足 LRU (最近最少使用) 缓存 约束的数据结构. 实现 LRUCache 类: LRUCache(int capacity) 以 正整数 作为容量 capacity ...
- 免费获取最新WebStorm激活码,永久激活WebStorm
在互联网上,目前还没有查询到一篇写得比较详细的WebStorm安装和激活教程.今天我将使用WebStorm最新2023年版本,从下载到安装以及创建项目带大家完整的走一遍. 分享的 WebStorm 2 ...
- 一天吃透Spring面试八股文
内容摘自我的学习网站:topjavaer.cn Spring是一个轻量级的开源开发框架,主要用于管理 Java 应用程序中的组件和对象,并提供各种服务,如事务管理.安全控制.面向切面编程和远程访问等. ...
- Linux目录结构及常用命令
目录 Linux目录结构... 1 Linux目录结构... 1 1. Linux常用命令... 4 1.1 Linux命令初体验... 4 1.2 文件目录操作命令... 8 1.3 拷贝移动命令. ...
- @Target元注解的使用
@Target注解标记另外的注解用于限制此注解可以应用哪种Java元素类型.先看Java SE 8中@Target是如何声明的: package java.lang.annotation; publi ...