题意:给你两个串A,B, 问一个串长为M+N且包含A和B且恰好包含M个R的字符串有多少种组合方式,所有字符串中均只含有字符L和R。

dp[i][j][k][S]表示串长为i,有j个R,在自动机中的状态为k,包含AB的状态为S的方案个数。

PS1.之前用long long int超时了两次

PS2.把行列搞错了WA了几次

#include <cstdio>
#include <cstdlib>
#include <algorithm>
#include <cstring> //#define LL long long int using namespace std; const int MAX_NODE = ;
const int CHILD_NUM = ;
const int MAXN = ;
const int MOD = ; struct ACAutomaton
{
int chd[MAX_NODE][CHILD_NUM]; //每个节点的儿子,即当前节点的状态转移
int val[MAX_NODE]; //记录题目给的关键数据
int fail[MAX_NODE]; //传说中的fail指针
int Q[MAX_NODE<<]; //队列,用于广度优先计算fail指针
int ID[]; //字母对应的ID
int sz; //已使用节点个数 //初始化,计算字母对应的儿子ID,如:'a'->0 ... 'z'->25
void Initialize()
{
fail[] = ;
ID['L'] = ;
ID['R'] = ;
return;
}
//重新建树需先Reset
void Reset()
{
memset(chd[] , , sizeof(chd[]));
val[] = ;
sz = ;
}
//将权值为key的字符串a插入到trie中
void Insert(char *a,int key)
{
int p = ;
for ( ; *a ; a ++)
{
int c = ID[*a];
if (!chd[p][c])
{
memset(chd[sz] , , sizeof(chd[sz]));
val[sz] = ;
chd[p][c] = sz ++;
}
p = chd[p][c];
}
val[p] = key;
}
//建立AC自动机,确定每个节点的权值以及状态转移
void Construct()
{
int *s = Q , *e = Q;
for (int i = ; i < CHILD_NUM ; i ++)
{
if (chd[][i])
{
fail[ chd[][i] ] = ;
*e ++ = chd[][i];
}
}
while (s != e)
{
int u = *s++;
for (int i = ; i < CHILD_NUM ; i ++)
{
int &v = chd[u][i];
if (v)
{
*e ++ = v;
fail[v] = chd[ fail[u] ][i];
//以下一行代码要根据题目所给val的含义来写
val[v] |= val[ fail[v] ];
}
else
{
v = chd[ fail[u] ][i];
}
}
}
}
} AC; int M, N;
char str[MAXN];
int dp[][MAXN][MAXN<<][<<]; void solved()
{
int pre = , cur = ; memset( dp[], , sizeof(dp[]) );
dp[][][][] = ;
int all = << ;
int L = M + N; for ( int i = ; i < L; ++i )
{
memset( dp[cur], , sizeof(dp[cur]) );
for ( int j = ; j <= M; ++j )
for ( int k = ; k < AC.sz; ++k )
for ( int r = ; r < ; ++r )
for ( int S = ; S < all; ++S )
{
int next = AC.chd[k][r];
dp[cur][j+r][next][ S|AC.val[next] ] += dp[pre][j][k][S];
dp[cur][j+r][next][ S|AC.val[next] ] %= MOD;
} cur ^= ;
pre ^= ;
} int ans = ;
for ( int j = ; j < AC.sz; ++j )
{
ans += dp[pre][M][j][all-];
ans %= MOD;
}
printf( "%d\n", ans ); return;
} int main()
{
AC.Initialize();
int T;
scanf( "%d", &T );
while ( T-- )
{
scanf( "%d%d", &M, &N );
AC.Reset();
for ( int i = ; i < ; ++i )
{
scanf( "%s", str );
AC.Insert( str, << i );
}
AC.Construct(); solved();
}
return ;
}

HDU 4758 Walk Through Squares( AC自动机 + 状态压缩DP )的更多相关文章

  1. HDU 4758 Walk Through Squares(AC自动机+DP)

    题目链接 难得出一个AC自动机,我还没做到这个题呢...这题思路不难想,小小的状压出一维来,不过,D和R,让我wa死了,AC自动机,还得刷啊... #include<iostream> # ...

  2. HDU 4057 Rescue the Rabbit ( AC自动机 + 状态压缩DP )

    模板来自notonlysuccess. 模式串只有10个,并且重复出现的分值不累加,因此很容易想到状态压缩. 将模式串加入AC自动机,最多有10*100个状态. dp[i][j][k]:串长为i,在T ...

  3. HDU - 4758 Walk Through Squares (AC自己主动机+DP)

    Description   On the beaming day of 60th anniversary of NJUST, as a military college which was Secon ...

  4. HDU 4511 (AC自动机+状态压缩DP)

    题目链接:  http://acm.hdu.edu.cn/showproblem.php?pid=4511 题目大意:从1走到N,中间可以选择性经过某些点,比如1->N,或1->2-> ...

  5. hdu 4057(ac自动机+状态压缩dp)

    题意:容易理解... 分析:题目中给的模式串的个数最多为10个,于是想到用状态压缩dp来做,它的状态范围为1-2^9,所以最大为2^10-1,那我们可以用:dp[i][j][k]表示长度为i,在tri ...

  6. POJ 3691 (AC自动机+状态压缩DP)

    题目链接:  http://poj.org/problem?id=3691 题目大意:给定N个致病DNA片段以及一个最终DNA片段.问最终DNA片段最少修改多少个字符,使得不包含任一致病DNA. 解题 ...

  7. bzoj1195 神奇的ac自动机+状态压缩dp

    /* 难的不是ac自动机,是状态压缩dp 之前做了一两题类似题目,感觉理解的还不够透彻 */ #include<iostream> #include<cstdio> #incl ...

  8. HDU 4758 Walk Through Squares ( Trie图 && 状压DP && 数量限制类型 )

    题意 : 给出一个 n 行.m 列的方格图,现从图左上角(0, 0) 到右下角的 (n, m)走出一个字符串(规定只能往下或者往右走),向右走代表' R ' 向下走则是代表 ' D ' 最后从左上角到 ...

  9. hdu 2825(ac自动机+状态压缩dp)

    题意:容易理解... 分析:在做这道题之前我做了hdu 4057,都是同一种类型的题,因为题中给的模式串的个数最多只能为10个,所以我们就很容易想到用状态压缩来做,但是开始的时候我的代码超时了dp时我 ...

随机推荐

  1. MyBatis中解决字段名与实体类属性名不相同的冲突

    一: 通过在查询的sql语句中定义字段名的别名,让字段名的别名和实体类的属性名一致,这样就可以表的字段名和实体类的属性名一一对应上了,这种方式是通过在sql语句中定义别名来解决字段名和属性名的映射关系 ...

  2. 手动创建maven项目+cmd+webapp+tomcat

    1.创建文件夹 2.在刚刚创建的文件夹目录下:创建一个scr文件夹和pom.xml文件 3.在pom.xml配置 <?xml version="1.0" encoding=& ...

  3. 3、SpringBoot+Mybatis整合------主键回填

    开发工具:STS 代码下载链接:https://github.com/theIndoorTrain/SpringBoot_Mybatis01/tree/d68efe51774fc4d96e5c6870 ...

  4. java XML 通过BeanUtils的population为对象赋值 根据用户选择进行dom4j解析

    根据xml文件设计Student对象 <?xml version="1.0" encoding="UTF-8"?> <students> ...

  5. ES6初识-Decorator

    开始先按照个插件 npm install babel-plugin-transform-decorators-lagacy --save-dev 1.扩充和修改类的行为 2.修改的行为@readonl ...

  6. LeetCode997. Find the Town Judge

    题目 在一个小镇里,按从 1 到 N 标记了 N 个人.传言称,这些人中有一个是小镇上的秘密法官. 如果小镇的法官真的存在,那么: 小镇的法官不相信任何人. 每个人(除了小镇法官外)都信任小镇的法官. ...

  7. BFS练习-POJ.2386

    Lake Counting Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 35122 Accepted: 17437 Descr ...

  8. Nginx+php+mysql+wordpress搭建自己的博客站点

    服务器环境要求Centos 6 或以上版本(由于我们的目标是半小时内搭建好,那就选简单yum安装)MySQL 5或更新版本Nginx 1或更新版本PHP 5 或更新版本 php-fpm 5或更新版本 ...

  9. Ubuntu16.04下配置ssh免密登录

    Ubuntu16.04下配置ssh免密登录 环境准备:新建两台虚拟机,而且两台虚拟机上都装有Ubuntu16.04的系统,使两台虚拟机之间保持互通状态.分别为两台虚拟机命名为A,B.假设我们要使A虚拟 ...

  10. PAT (Basic Level) Practice 1023 组个最小数

    个人练习 给定数字 0-9 各若干个.你可以以任意顺序排列这些数字,但必须全部使用.目标是使得最后得到的数尽可能小(注意 0 不能做首位).例如:给定两个 0,两个 1,三个 5,一个 8,我们得到的 ...