【问题描述】

hy 抄题解又被老师抓住了,现在老师把他叫到了办公室。 老师要 hy 和他玩一个游

戏。如果 hy 输了,老师就要把他开除信息组;

游戏分为 k 轮。在游戏开始之前,老师会将 n 个由英文字母组成的字符串放入箱子。

每局开始,字符串为空串,然后两人轮流在末尾追加字符,保证新的字符串为箱子中某

字符串的前缀,直到有一个人不能操作,不能操作的那个人就输掉当前的一轮。新一轮

由上一句输的人先手。最后一局赢的人获胜。

假定老师和 hy 都能采取最优的策略,且老师为了彰显自己的大度让 hy 先手,求 hy 能否

获胜。

【输入格式】

输入包括多组数据,输入以文字流结尾(EOF)为结束。

每组数据的第一行包含两个整数 n, k,分别表示放入箱子字符串的数量和游戏的轮

数。

接下来 n 行,每行一个字符串表示由英文字母组成的句子。

【输出格式】

每组数据第一行,输出 hy 是否能赢,若能赢输出”HY wins!“,否则输出”Teacher

wins!”。

【样例输入 1】

2 3

a

b

3 1

a

b

c

【样例输出 1】

HY wins!

HY wins!

【样例输入 2】

1 2

ab

【样例输出 2】

Teacher wins!【评测用例规模与约定】

对于 40%的评测用例,1≤n≤10,1≤k≤10 4 ;

对于 100%的评测用例,1≤n≤10 5 ,1≤k≤10 9 ,保证所有字符串总长度不超过 10 5 ,数据组

数不超过 10

Solution

先把所有的字符串插入 Trie 树,然后就是博弈论了。

博弈论分为三种情况:

1.若先手无必胜策略即先手必败则先手一直先手,最后一局后手胜;

2.若先手有必胜策略则下一局成后手,即为胜败交替,此时,最后一句的胜败决定于 k 的奇

偶性;

3.先手有必胜策略有必败策略,则先手前 k-1 局败,最后一局先手胜。

考虑 \(dp\) 转移 ,\(f[i]\) 代表当前节点是否能肯定赢,\(F[i]\) 代表当前节点是否能肯定输。

  • 对于深度为偶数的节点,那么只要子节点中有任意一个满足,即可使当前 \(f[x]\) 或者 \(F[x]\) 为 \(1\) 。

    因为此时为 HY 做决定的时间。
  • 对于深度为奇数的节点,即此时由 Teacher 做决定,那么此时只有当其所有子节点都为肯定输或者肯定赢是才能计数。

然后按部就班转移即可。

Code

#include<bits/stdc++.h>
#define N 100001
#define ll long long
using namespace std; int ch[N][26],cnt;
void insert(string s)
{
int u=0,n=s.length();
for(int i=0;i<n;i++)
{
if(!ch[u][s[i]-'a'])
ch[u][s[i]-'a']=++cnt;
u=ch[u][s[i]-'a'];
} return;
} int pd[N],Pd[N],sum[N];
void dfs(int x,int dep,int zm)
{
sum[x]=1;
for(int i=0;i<26;i++)
if(ch[x][i])
{
dfs(ch[x][i],dep+1,i);
sum[x]+=sum[ch[x][i]];
}
if(sum[x]==1)
{
if(dep%2==1)pd[x]=1;
else Pd[x]=1;
return;
}
if(dep%2!=1)
{
for(int i=0;i<26;i++)
if(ch[x][i])
{
pd[x]=max(pd[x],pd[ch[x][i]]);
Pd[x]=max(Pd[x],Pd[ch[x][i]]);
}
}else
{
int pp=1,qq=1;
for(int i=0;i<26;i++)
if(ch[x][i])
{
if(!pd[ch[x][i]])pp=0;
if(!Pd[ch[x][i]])qq=0;
}pd[x]=pp,Pd[x]=qq;
}
return;
} int n,k;
int main()
{
freopen("amerce.in","r",stdin);
freopen("amerce.out","w",stdout);
while(scanf("%d",&n)!=EOF)
{
memset(pd,0,sizeof(pd));
memset(Pd,0,sizeof(Pd));
memset(sum,0,sizeof(sum));
memset(ch,0,sizeof(ch));
scanf("%d",&k); cnt=0;
for(int i=1;i<=n;i++)
{
string s;
cin>>s; insert(s);
}
dfs(0,0,0);
if(pd[0]!=1){cout<<"Teacher wins!"<<endl;continue;}
if(pd[0]==1&&Pd[0]==0)
{
if(k%2==1)cout<<"HY wins!"<<endl;
else cout<<"Teacher wins!"<<endl;
continue;
}
if(pd[0]==1&&Pd[0]==1)
cout<<"HY wins!"<<endl;
}
return 0;
}

HY 的惩罚 (Trie 树,博弈论)的更多相关文章

  1. SPOJ11414 COT3 博弈论 + Trie树合并

    考虑对于每个子树从下往上依次考虑 对于叶子节点而言,如果可以染色,那么其\(sg\)值为\(1\),否则为\(0\) 考虑往上合并 如果选择了\(x\),那么后继状态就是其所有子树 如果选了其他子树中 ...

  2. 数据结构 | 30行代码,手把手带你实现Trie树

    本文始发于个人公众号:TechFlow,原创不易,求个关注 今天是算法和数据结构专题的第28篇文章,我们一起来聊聊一个经典的字符串处理数据结构--Trie. 在之前的4篇文章当中我们介绍了关于博弈论的 ...

  3. 基于trie树做一个ac自动机

    基于trie树做一个ac自动机 #!/usr/bin/python # -*- coding: utf-8 -*- class Node: def __init__(self): self.value ...

  4. 基于trie树的具有联想功能的文本编辑器

    之前的软件设计与开发实践课程中,自己构思的大作业题目.做的具有核心功能,但是还欠缺边边角角的小功能和持久化数据结构,先放出来,有机会一点点改.github:https://github.com/chu ...

  5. hihocoder-1014 Trie树

    hihocoder 1014 : Trie树 link: https://hihocoder.com/problemset/problem/1014 题意: 实现Trie树,实现对单词的快速统计. # ...

  6. 洛谷P2412 查单词 [trie树 RMQ]

    题目背景 滚粗了的HansBug在收拾旧英语书,然而他发现了什么奇妙的东西. 题目描述 udp2.T3如果遇到相同的字符串,输出后面的 蒟蒻HansBug在一本英语书里面找到了一个单词表,包含N个单词 ...

  7. 通过trie树实现单词自动补全

    /** * 实现单词补全功能 */ #include <stdio.h> #include <stdlib.h> #include <string.h> #incl ...

  8. #1014 Trie树

    本题主要是求构造一棵Trie树,即词典树用于统计单词. C#代码如下: using System; using System.Collections.Generic; using System.Lin ...

  9. Trie树-字典查找

    描述 小Hi和小Ho是一对好朋友,出生在信息化社会的他们对编程产生了莫大的兴趣,他们约定好互相帮助,在编程的学习道路上一同前进. 这一天,他们遇到了一本词典,于是小Hi就向小Ho提出了那个经典的问题: ...

随机推荐

  1. 阶段1 语言基础+高级_1-3-Java语言高级_1-常用API_1_第3节 Random类_9-生成指定范围的随机数

    左闭右开区间

  2. pyenv python 版本控制

    经常遇到这样的情况: 系统自带的Python是2.x,自己需要Python 3.x,此时需要在系统中安装多个Python,但又不能影响系统自带的Python,即需要实现Python的多版本共存,pye ...

  3. Oracle的substr函数简单用法(转)

    转:http://www.cnblogs.com/nicholas_f/articles/1526063.html substr(字符串,截取开始位置,截取长度) //返回截取的字 substr('H ...

  4. [19/05/07-星期二] JDBC(Java DataBase Connectivity)_CLOB(存储大量的文本数据)与BLOB(存储大量的二进制数据)

    一. CLOB(Character Large Object ) – 用于存储大量的文本数据 – 大字段有些特殊,不同数据库处理的方式不一样,大字段的操作常常是以流的方式来处理的.而非一般的字段,一次 ...

  5. 洛谷 P1462 通往奥格瑞玛的道路(二分答案,堆优化dijkstra)

    传送门 解题思路 首先看题目问题,求经过的所有城市中最多的一次收取的费用的最小值是多少.一看“最大值最小”就想到了二分答案. 在读一遍题目,就是二分收取的费用,然后对于每一个二分的费用,跑一边最短路, ...

  6. Python之字符串和正则表达式

    使用正则表达式 Python对正则表达式的支持 例子:替换字符串中的不良内容 import re def main(): sentence = '你丫是傻叉吗? 我操你大爷的. Fuck you.' ...

  7. Gorgeous Sequence(线段树)

    Gorgeous Sequence Time Limit: 6000/3000 MS (Java/Others)    Memory Limit: 131072/131072 K (Java/Othe ...

  8. 问题 D: 小k的硬币问题

    问题 D: 小k的硬币问题 时间限制: 1 Sec  内存限制: 128 MB提交: 21  解决: 5[提交] [状态] [命题人:jsu_admin] 题目描述 小k和小p一起玩一个游戏,有n堆硬 ...

  9. Flutter 初探 -

    flutter 安装 经过许久的关注,及最近google算是真正地推行flutter时,加上掘金小册也有相应的教程,我知道自己得跟着这一波潮流学习了,不然迟早会面临着小程序的危(大家都会了就你不会), ...

  10. 基于 Python 的自定义分页组件

    基于 Python 的自定义分页组件 分页是网页中经常用到的地方,所以将分页功能分出来,作为一个组件可以方便地使用. 分页实际上就是不同的 url ,通过这些 url 获取不同的数据. 业务逻辑简介 ...