题意

给一个字符串\(s\),和\(n\)个子串\(t[i]\),两个人博弈,每次取出一个串\(t[i]\),在后面加入一个字符,保证新字符串仍然是\(s\)的子串,无法操作的人输。

分析

  • n个子串,类比于n堆石子,如果把子串\(t[i]\)在后面加若干字符能生成的子串看出一个状态,用一个数表示,那每次状态的变化,就类比于对一堆石子取走若干个,无法操作,就类比于没有石子可以取。
  • 多堆取石子游戏的做法就是把每堆石子的SG值(sg(x)=x)异或起来,不为零就先手赢,否则后手赢。
  • 所以可以将题目转化为求每个子串\(t[i]\)对应状态的SG值。
  • 然后就是后缀自动机的套路,每个子串就可以对应SAM上的一个节点,所以可以直接dfs记忆化搜索SG值。

代码

#include <bits/stdc++.h>
using namespace std;
const int N=1e5+50;
char s[N];
int n;
struct SAM{
int nex[N*2][26],fa[N*2],len[N*2],num[N*2];
int cnt,lst;
int sg[N*2];
int newnode(int l,int s){
for(int i=0;i<26;i++){
nex[cnt][i]=0;
}
len[cnt]=l;
num[cnt]=s;
return cnt++;
}
void init(){
cnt=0;
lst=newnode(0,0);
fa[lst]=-1;
memset(sg,-1,sizeof(sg));
}
void add(int c){
c-='a';
int p=lst;
int cur=newnode(len[p]+1,1);
while(p!=-1 && !nex[p][c]){
nex[p][c]=cur;
p=fa[p];
}
if(p==-1){
fa[cur]=0;
}else{
int q=nex[p][c];
if(len[q]==len[p]+1){
fa[cur]=q;
}else{
int cl=newnode(len[p]+1,0);
fa[cl]=fa[q];
memcpy(nex[cl],nex[q],sizeof(nex[cl]));
while(p!=-1 && nex[p][c]==q){
nex[p][c]=cl;
p=fa[p];
}
fa[q]=fa[cur]=cl;
}
}
lst=cur;
}
int w[N*2],tp[N*2];
void dfs(int u){
int vis[30]={0};
if(sg[u]!=-1){
return;
}
for(int i=0;i<26;i++){
int v=nex[u][i];
if(v){
dfs(v);
vis[sg[v]]=1;
}
}
for(int i=0;i<26;i++){
if(!vis[i]){
sg[u]=i;
break;
}
}
}
void sfd(int l){
for(int i=0;i<=l;i++){
w[i]=0;
}
for(int i=1;i<=cnt;i++){
w[len[i]]++;
}
for(int i=2;i<=l;i++){
w[i]+=w[i-1];
}
for(int i=cnt-1;i>=1;i--){
tp[w[len[i]]--]=i;
}
sg[lst]=0;
int vis[30]={0};
for(int i=cnt-1;i>=0;i--){
int u=tp[i];
memset(vis,0,sizeof(vis));
for(int j=0;j<26;j++){
int v=nex[u][j];
if(v){
vis[sg[v]]=1;
}
}
for(int j=0;j<26;j++){
if(!vis[j]){
sg[u]=j;
break;
}
}
}
}
int solve(char *s){
int rt=0;
int l=strlen(s);
for(int j=0;j<l;j++){
rt=nex[rt][s[j]-'a'];
}
return sg[rt];
}
}ac;
int main(){
// freopen("in.txt","r",stdin);
while(~scanf("%s",s)){
ac.init();
int len=strlen(s);
for(int i=0;i<len;i++){
ac.add(s[i]);
}
//dfs或者拓扑排序求sg函数
// ac.dfs(0);
ac.sfd(len);
scanf("%d",&n);
int ans=0;
for(int i=1;i<=n;i++){
scanf("%s",s);
ans^=ac.solve(s);
}
if(ans){
printf("Alice\n");
}else{
printf("Bob\n");
}
}
return 0;
}

是男人就过八题A_A String Game题解的更多相关文章

  1. poj 1741 楼教主男人八题之中的一个:树分治

    http://poj.org/problem? id=1741 Description Give a tree with n vertices,each edge has a length(posit ...

  2. poj 1737男人八题之一 orz ltc

    这是楼教主的男人八题之一.很高兴我能做八分之一的男人了. 题目大意:求有n个顶点的连通图有多少个. 解法: 1.  用总数减去不联通的图(网上说可以,我觉得时间悬) 2.    用动态规划(数学递推) ...

  3. POJ1742 Coins(男人八题之一)

    前言 大名鼎鼎的男人八题,终于见识了... 题面 http://poj.org/problem?id=1742 分析 § 1 多重背包 这很显然是一个完全背包问题,考虑转移方程: DP[i][j]表示 ...

  4. Cogs 1714. [POJ1741][男人八题]树上的点对(点分治)

    [POJ1741][男人八题]树上的点对 ★★★ 输入文件:poj1741_tree.in 输出文件:poj1741_tree.out 简单对比 时间限制:1 s 内存限制:256 MB [题目描述] ...

  5. 《学习OpenCV》练习题第四章第八题ab

    这道题是利用OpenCV例子程序里自带的人脸检测程序,做点图像的复制操作以及alpha融合. 说明:人脸检测的程序我参照了网上现有的例子程序,没有用我用的OpenCV版本(2.4.5)的facedet ...

  6. 经典算法题每日演练——第八题 AC自动机

    原文:经典算法题每日演练--第八题 AC自动机 上一篇我们说了单模式匹配算法KMP,现在我们有需求了,我要检查一篇文章中是否有某些敏感词,这其实就是多模式匹配的问题. 当然你也可以用KMP算法求出,那 ...

  7. 【20171026早】alert(1) to win - 第六、七、八题

    早上7点起床,又写了一篇小说发在了起点网上,有兴趣的可以看看.点击这里 忙完后,继续练习,刚开始发现自己答题的速度有些慢,可能是因为对于html,javascript知识不是很精通,但是话又说回来,谁 ...

  8. [是男人就过8题——Pony.ai]Perfect N-P Arrays

    [是男人就过8题--Pony.ai]Perfect N-P Arrays 题目大意: 一棵\(n(\sum n\le5\times10^6)\)个结点的树,每个结点都有一个括号.求树上一个合法的括号序 ...

  9. noip做题记录+挑战一句话题解?

    因为灵巧实在太弱辽不得不做点noip续下命QQAQQQ 2018 积木大赛/铺设道路 傻逼原题? 然后傻逼的我居然检查了半天是不是有陷阱最后花了差不多一个小时才做掉我做过的原题...真的傻逼了我:( ...

随机推荐

  1. codeforces 389 D. Fox and Minimal path(构造+思维)

    题目链接:https://vjudge.net/contest/175446#problem/J 题解:显然要用最多n个点构成的图要使的得到的最短路条数有1e9次个,显然要有几个数相乘容易想到2的几进 ...

  2. hdu 4614 Vases and Flowers(线段树)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4614 题意: 给你N个花瓶,编号是0  到 N - 1 ,初始状态花瓶是空的,每个花瓶最多插一朵花. ...

  3. codeforces 361 D. Levko and Array(dp+二分)

    题目链接:http://codeforces.com/contest/361/problem/D 题意:最多可以修改K次数字,每次修改一个数字变成任意值,C=max(a[i+1]-a[i]):求操作之 ...

  4. vs 模板更新

    vs 模板更新,执行命令: dotnet new --install McMaster.DotNet.GlobalTool.Templates

  5. win、mac 设置 php上传文件大小限制

    修改php.ini win平台WAMP修改 步骤 左键点击wamp 选择php 在弹出的窗口中选择php.ini 在打开的文件中进行修改(修改步骤如下) 修改完毕,保存并重启wamp mac  MAM ...

  6. Java的异常:Error与Exception

    一. 异常机制的概述      异常机制是指当程序出现错误后,程序如何处理.具体来说,异常机制提供了程序退出的安全通道.当出现错误后,程序执行的流程发生改变,程序的控制权转移到异常处理器. 程序错误分 ...

  7. springboot使用memcache缓存

    Memcached简介 Memcached 是一个高性能的分布式内存对象缓存系统,用于动态Web应用以减轻数据库负载.它通过在内存中缓存数据和对象来减少读取数据库的次数,从而提高动态.数据库驱动网站的 ...

  8. C# 表达式树讲解(一)

    一.前言 一直想写一篇Dpper的定制化扩展的文章,但是里面会设计到对Lambda表达式的解析,而解析Lambda表达式,就必须要知道表达式树的相关知识点.我希望能通过对各个模块的知识点或者运用能够多 ...

  9. 重学Java(一):与《Java编程思想》的不解之缘

    说起来非常惭愧,我在 2008 年的时候就接触了 Java,但一直到现在(2018 年 10 月 10 日),基础知识依然非常薄弱.用一句话自嘲就是:十年 IT 老兵,Java 菜鸡一枚. 于是,我想 ...

  10. java 手机号码+邮箱的验证

    import java.util.regex.Pattern; //导入的包 1:String REGEX_MOBILE = "^((17[0-9])|(14[0-9])|(13[0-9]) ...