题意

给一个字符串\(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. HTML(五)列表,区块,布局,表单和输入

    HTML 列表 无序列表 Coffee Tea Milk 默认是圆点,也可以 圆圈 正方形 有序列表 Coffee Tea Milk Coffee Tea Milk 默认是用数字排序 大写字母 小写字 ...

  2. 牛客-2018多校算法第五场D-集合问题+并查集

    集合问题 题意: 给你a,b和n个数p[i],问你如何分配这n个数给A,B集合,并且满足: 若x在集合A中,则a-x必须也在集合A中. 若x在集合B中,则b-x必须也在集合B中. 思路:并查集操作,自 ...

  3. 【百度之星】【java大数+C++做法】hdu 6719 Strassen

    代码:递归搜索一下.java大数做法 import java.util.*; import java.math.*; import java.security.MessageDigest; publi ...

  4. yzoj P2045 不老的传说 题解

    题意 类似于涂色问题,环形问题不多说,断环加倍即可,限制条件if判断就行. 解析 代码 #include<bits/stdc++.h> using namespace std; int n ...

  5. spring boot监听器的实现

    spring boot监听器的实现 如下所示: import javax.servlet.ServletContextEvent; import javax.servlet.ServletContex ...

  6. Java反序列化漏洞原理解析(案例未完善后续补充)

    序列化与反序列化 序列化用途:方便于对象在网络中的传输和存储 java的反序列化 序列化就是将对象转换为流,利于储存和传输的格式 反序列化与序列化相反,将流转换为对象 例如:json序列化.XML序列 ...

  7. sqoop导oracle数据到hive中并动态分区

    静态分区: 在hive中创建表可以使用hql脚本: test.hql USE TEST; CREATE TABLE page_view(viewTime INT, userid BIGINT, pag ...

  8. 【学习笔记】第七章 python3核心技术与实践--输入与输出

    [第六章]思考题答案,仅供参考: # coding:utf-8import time#方法一start_time = time.perf_counter()s = ''for n in range(0 ...

  9. All Of ACM

    数据结构和算法专栏,我会什么写什么  = = 不定时更新 一.数据结构 树状数组详解 线段树详解 二.算法 KMP算法 三.板子 我的代码模板 大整数模板 四.题解报告 几道STL题目(FJUT - ...

  10. Linux环境基于CentOS7 搭建部署Docker容器

    1.Docker容器概述 区分Docker容器技术和VM虚拟机技术: evernotecid://394EFE90-9CE0-4D65-A8CD-DFEC0DC8061E/appyinxiangcom ...