题意

给一个字符串\(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. hdu-6644 11 Dimensions

    题目链接 11 Dimensions Problem Description 11 Dimensions is a cute contestant being talented in math. On ...

  2. 洛谷P2331 [SCOI2005]最大子矩阵 DP

    P2331 [SCOI2005]最大子矩阵 题意 : 这里有一个n*m的矩阵,请你选出其中k个子矩阵,使得这个k个子矩阵分值之和最大.注意:选出的k个子矩阵不能相互重叠. 第一行为n,m,k(1≤n≤ ...

  3. 2019 Multi-University Training Contest 7

    2019 Multi-University Training Contest 7 A. A + B = C 题意 给出 \(a,b,c\) 解方程 \(a10^x+b10^y=c10^z\). tri ...

  4. 数论 线性同余方程的应用 poj2891

    Strange Way to Express Integers Time Limit: 1000MS   Memory Limit: 131072K Total Submissions: 17321 ...

  5. 百度之星 资格赛 1003 度度熊与邪恶大魔王 dp(背包)

    度度熊与邪恶大魔王  Accepts: 1141  Submissions: 6840  Time Limit: 2000/1000 MS (Java/Others)  Memory Limit: 3 ...

  6. 你知道@RequestMapping的name属性有什么用吗?【享学Spring MVC】

    每篇一句 牛逼架构师:把复杂问题简单化,把简单问题搞没 菜逼架构师:把简单问题复杂化 前言 不知这个标题能否勾起你的好奇心和求知欲?在Spring MVC的使用中,若我说@RequestMapping ...

  7. js-DOM ~ 05. Date日期的相关操作、string、查字符串的位置、给索引查字符、字符串截取slice/substr/substring、去除空格、替换、大小写、Math函数、事件绑定、this

    内置对象:  语言自带的对象/提供了常用的.基本的功能 打印数组和字符串不用for... in   /   打印josn的时候采用for...in Date 获取当前事件:   var date = ...

  8. Maven工程读取properties文件过程

    1.创建需要读取的properties文件 2.在xml文件中加载配置文件 <!-- 加载配置文件 --> <context:property-placeholder locatio ...

  9. a417: 螺旋矩陣

    题目: 每行有一正整数T,代表有几组测试数据 接下来有T行,每行有N.M两正整数 N为矩阵长宽,就是会有N*N矩阵 M为方向,M=1为顺时钟,M=2为逆时钟 N范围为1~100之间 思路: 所以,代码 ...

  10. python+selenium十:selenium的二次封装

    python+selenium十:基于原生selenium的二次封装   from selenium import webdriverfrom selenium.webdriver.support.w ...