雅礼集训【Day6-1】字符串

假设我们有串\(a\),我们设\(a'\)为\(a\)翻转后按为取反过后的串。

我们只考虑前一半的,长为\(m\)的串。如果前半截匹配了\(a\)或者\(a'\),则\(a\)就被匹配上了。所以我们记\(f_{i,j,S}\)表示长度\(i\),在AC自动机上匹配到了\(j\)节点,已经匹配了的串的集合为\(S\)的方案数。

但是可能会出现\(a\)出现的位置跨越了\(m\),这样我们就会出问题。因为我们记录了生成的串在AC自动机上匹配的节点,所以我们就能得到\(a\)在前半截中匹配的长度。如果这个长度\(\geq\lfloor \frac{len}{2}\rfloor\),则我们能知道后半截是什么,也就能判断是否合法了。

那如果\(a\)没有匹配长度\(\geq\lfloor \frac{len}{2}\rfloor\)的情况呢?我们发现,此时\(a'\)的匹配长度一定\(\geq\lfloor \frac{len}{2}\rfloor\),所以我们同时判断\(a\)和\(a'\),其中一个成立就行了。

代码:

#include<bits/stdc++.h>
#define ll long long
#define N 2005
#define M 505 using namespace std;
inline int Get() {int x=0,f=1;char ch=getchar();while(ch<'0'||ch>'9') {if(ch=='-') f=-1;ch=getchar();}while('0'<=ch&&ch<='9') {x=(x<<1)+(x<<3)+ch-'0';ch=getchar();}return x*f;} const ll mod=998244353;
int n,m;
string str[20];
struct trie {
int ch[2];
int w;
int fail;
int flag;
int len;
}tr[N]; int rt=1;
int cnt=1;
void Insert(string s,int id) {
int len=s.length();
int v=1;
for(int i=0;i<len;i++) {
int j=s[i]-'0';
if(!tr[v].ch[j]) {
tr[v].ch[j]=++cnt;
tr[tr[v].ch[j]].len=tr[v].len+1;
}
v=tr[v].ch[j];
}
tr[v].w|=1<<id-1;
} queue<int>q;
void build_fail() {
for(int i=0;i<2;i++) {
if(!tr[1].ch[i]) tr[1].ch[i]=1;
else {
int x=tr[1].ch[i];
tr[x].fail=1;
q.push(x);
}
}
while(!q.empty()) {
int v=q.front();
q.pop();
tr[v].w|=tr[tr[v].fail].w;
for(int i=0;i<2;i++) {
if(!tr[v].ch[i]) tr[v].ch[i]=tr[tr[v].fail].ch[i];
else {
int sn=tr[v].ch[i];
int f=tr[v].fail;
tr[sn].fail=tr[f].ch[i];
q.push(sn);
}
}
}
}
int pre[20][N];
int f[M][1205][1<<6];
bool pd(string &a,int len,int v,int id) {
int now=1;
tr[1].flag=1;
for(int i=0;i<len;i++) {
now=tr[now].ch[a[i]-'0'];
tr[now].flag=1;
}
while(!tr[v].flag) v=tr[v].fail;
int nlen=0;
now=1;
for(int i=0;i<len&&now!=v;i++,nlen++) {
now=tr[now].ch[a[i]-'0'];
}
int flag=0;
while(v) {
if(tr[v].flag&&pre[id][tr[v].len]) flag=1;
v=tr[v].fail;
}
tr[1].flag=0;
now=1;
for(int i=0;i<len;i++) {
now=tr[now].ch[a[i]-'0'];
tr[now].flag=0;
}
return flag;
} bool chk(int v,int S) {
for(int i=1;i<=n;i++) {
if(S>>i-1&1) continue ;
int len=str[i].length();
if(!pd(str[i],len,v,i)&&!pd(str[i+n],len,v,i+n)) return 0;
}
return 1;
} int main() {
n=Get(),m=Get();
for(int i=1;i<=n;i++) cin>>str[i];
for(int i=1;i<=n;i++) {
int len=str[i].length();
for(int j=len-1;j>=0;j--) {
str[i+n].push_back(str[i][j]^1);
}
}
for(int i=1;i<=n;i++) {
Insert(str[i],i);
Insert(str[i+n],i);
}
build_fail();
f[0][1][0]=1;
for(int i=0;i<m;i++) {
for(int j=1;j<=cnt;j++) {
for(int S=0;S<1<<n;S++) {
if(!f[i][j][S]) continue ;
for(int k=0;k<2;k++) {
int to=tr[j].ch[k];
(f[i+1][to][S|tr[to].w]+=f[i][j][S])%=mod;
}
}
}
} for(int i=1;i<=2*n;i++) {
int len=str[i].length();
for(int st=1;st<=len;st++) {
if(st*2<len) continue ;
int flag=1;
for(int j=st;j<len;j++) {
if(str[i][j]==str[i][2*st-j-1]) flag=0;
}
pre[i][st]=flag;
}
} int ans=0;
for(int i=1;i<=cnt;i++) {
for(int S=0;S<1<<n;S++) {
if(!f[m][i][S]) continue ;
if(chk(i,S)) {
(ans+=f[m][i][S])%=mod;
}
}
}
cout<<ans;
return 0;
}

雅礼集训【Day6-1】字符串的更多相关文章

  1. 雅礼集训 Day6 T2 Equation 解题报告

    Equation 题目描述 有一棵\(n\)个点的以\(1\)为根的树,以及\(n\)个整数变量\(x_i\).树上\(i\)的父亲是\(f_i\),每条边\((i,f_i)\)有一个权值\(w_i\ ...

  2. 雅礼集训 Day6 T1 Merchant 解题报告

    Merchant 题目描述 有\(n\)个物品,第\(i\)个物品有两个属性\(k_i,b_i\),表示它在时刻\(x\)的价值为\(k_i\times x+b_i\). 当前处于时刻\(0\),你可 ...

  3. [LOJ 6031]「雅礼集训 2017 Day1」字符串

    [LOJ 6031] 「雅礼集训 2017 Day1」字符串 题意 给定一个长度为 \(n\) 的字符串 \(s\), \(m\) 对 \((l_i,r_i)\), 回答 \(q\) 个询问. 每个询 ...

  4. 「雅礼集训 2017 Day7」事情的相似度

    「雅礼集训 2017 Day7」事情的相似度 题目链接 我们先将字符串建后缀自动机.然后对于两个前缀\([1,i]\),\([1,j]\),他们的最长公共后缀长度就是他们在\(fail\)树上对应节点 ...

  5. 雅礼集训1-9day爆零记

    雅礼集训1-9day爆零记 先膜一下虐爆我的JEFF巨佬 Day0 我也不知道我要去干嘛,就不想搞文化科 (文化太辣鸡了.jpg) 听李总说可以去看(羡慕)各路大佬谈笑风声,我就报一个名吧,没想到还真 ...

  6. LOJ_6045_「雅礼集训 2017 Day8」价 _最小割

    LOJ_6045_「雅礼集训 2017 Day8」价 _最小割 描述: 有$n$种减肥药,$n$种药材,每种减肥药有一些对应的药材和一个收益. 假设选择吃下$K$种减肥药,那么需要这$K$种减肥药包含 ...

  7. 「雅礼集训 2017 Day2」解题报告

    「雅礼集训 2017 Day2」水箱 我怎么知道这种题目都能构造树形结构. 根据高度构造一棵树,在树上倍增找到最大的小于约束条件高度的隔板,开一个 \(vector\) 记录一下,然后对于每个 \(v ...

  8. 「雅礼集训 2017 Day1」 解题报告

    「雅礼集训 2017 Day1」市场 挺神仙的一题.涉及区间加.区间除.区间最小值和区间和.虽然标算就是暴力,但是复杂度是有保证的. 我们知道如果线段树上的一个结点,\(max=min\) 或者 \( ...

  9. [LOJ 6030]「雅礼集训 2017 Day1」矩阵

    [LOJ 6030] 「雅礼集训 2017 Day1」矩阵 题意 给定一个 \(n\times n\) 的 01 矩阵, 每次操作可以将一行转置后赋值给某一列, 问最少几次操作能让矩阵全为 1. 无解 ...

随机推荐

  1. Tomcat日志设定

    1    Tomcat 日志概述 Tomcat 日志信息分 为 两 类 : 一.是运行中的日志,它主要 记录 运行的一些信息,尤其是一些异常 错误 日志信息 .二.是 访问 日志信息,它 记录 的 访 ...

  2. Git中的"pull request"真正比较的是什么?

    前言 利用git版本控制工具时,我们通常会从主分支拉出新分支进行开发,开发完成后创建pr(也就是pull request),让其他小伙伴帮忙review,确定代码没有问题后再将新分支合并到主分支上.但 ...

  3. python中的魔法属性

    目录 1. __doc__ 2. __module__ 和 __class__ 3. __init__ 4. __del__ 5. __call__ 6. __dict__ 7. __str__ 8. ...

  4. Gvim 和vim 有什么区别

    Gvim 和vim 有什么区别 Gvim是windows的 vim是linux的黑色的命令符 Gvim是单独的窗口下的vim,像notepad一样. vim就是在黑乎乎的cmd窗口下的编辑器.wind ...

  5. Android LiveData使用

    LiveData是一个可观察的数据持有者类. 与常规observable不同,LiveData是生命周期感知的,当生命周期处于STARTED或RESUMED状态,则LiveData会将其视为活动状态, ...

  6. LNMP环境下安装Redis,以及php的redis扩展

    1.下载 sudo wget http://download.redis.io/releases/redis-4.0.9.tar.gz 2.解压 sudo tar zvxf redis-4.0.9.t ...

  7. 开源项目商业模式分析(2) - 持续维护的重要性 - Selenium和WatiN

    该系列第一篇发布后收到不少反馈,包括: 第一篇里说的MonicaHQ不一定盈利 没错,但是问题在于绝大多数开源项目商业数据并没有公开,从而无法判断其具体是否盈利.难得MonicaHQ是公开的,所以才用 ...

  8. 简单选择排序算法的C++实现

    简单选择排序采用最简单的选择方法,即在剩余序列中选出最小(或最大)的关键字,和剩余序列的第一个关键字交换位置,依次选择下去,直至使整个序列有序. 算法中两层循环的执行次数和初始序列没有关系,第二层循环 ...

  9. django rest framework 项目创建

    Django Rest Framework 是一个强大且灵活的工具包,用以构建Web API 为什么要使用Rest Framework Django REST Framework可以在Django的基 ...

  10. (后端)maven仓库

    仓库网址:http://mvnrepository.com/artifact/org.springframework/spring-core 可以去选择评分高的jar,复制: <!-- http ...