与其说这是一次重温AC自动机+dp,倒不如说这是个坑,而且把队友给深坑了。

这个题目都没A得出来,我只觉得我以前的AC自动机的题目都白刷了——深坑啊。

题目的意思是给你两个串,每个串只含有R或者D,要求有多少种长为(n+m)的串(其中有n个R,m个D)同时含有这两个串作为子串。(我就不说题目描述了)

看完了题目我居然一开始用排列组合去做,搞了近一个小时,深坑队友啊。

其实正解是用AC自动机来dp出解的。

什么意思呢?把两个给定的串放到AC字典树上,并且弄好fail指针,变为tire图,构造好AC自动机,然后就是dp了!

怎么dp呢?

可以这样来做,f[x][y][z][k]表示在第x个点,走了y个R,z个D,含有串的情况为k的种类数。

xyz就不用解释了,一看就懂,k是怎么表示状态数的呢?1表示有1个串,2表示两个串?显然不行,到时候有重复的你怎么判断啊 ?

这里k其实是用二进制来表示的,由于每次只有两个串,所有我们用00表示都不含,01表示含有第一个串,10表示含有第二个串,11表示两串都含有。

这样k就只要4就可以了。;

接下来就是自动机和dp的知识了。

我的这个代码写得很冗,效率灰常地低,希望大家不要见怪,我也是为了保险起见。

一开始我还M了好多发,注意这个题目一般至少要32M的内存才能过,hdu上那些900多k的代码我不知道是怎么来的。所以对我来说这个题目还有一点点卡内存的赶脚。

诶不说了说多了都是泪,上代码:

#include <iostream>
#include <cstdio>
#include <cstring>
#include <queue>
#define maxn 201
#define maxm 101
#define M 1000000007
using namespace std; int next[maxn][2],f[maxn][maxm][maxm][4],fail[maxn],N,tag[maxn],t,n,m;
char s0[maxn]; void clear()
{
N=0;
memset(next,0,sizeof next);
memset(f,0,sizeof f);
memset(fail,0,sizeof fail);
memset(tag,0,sizeof tag);
} void insert(char s[],int code)
{
int tep,now=0;
for (int i=0; s[i]; i++)
{
if (s[i]=='R') tep=0; else tep=1;
if (!next[now][tep]) next[now][tep]=++N;
now=next[now][tep];
}
tag[now]=(1<<(code-1));
} void buildAC()
{
queue<int>Q;
int now,k,child;
Q.push(0);
while (!Q.empty())
{
now=Q.front(),Q.pop();
for (int i=0; i<2; i++)
{
child=next[now][i];
if (child)
{
Q.push(child);
if (now==0) fail[child]=0;
else
{
k=fail[now];
while (k && !next[k][i]) k=fail[k];
if (next[k][i]) fail[child]=next[k][i];
else fail[child]=0;
}
tag[child]|=tag[fail[child]];
}
else
{
k=fail[now];
next[now][i]=next[k][i];
}
}
}
} int main()
{
int cur,num,ans;
scanf("%d",&t);
while (t--)
{
clear();
scanf("%d%d",&n,&m);
scanf("%s",s0); insert(s0,1);
scanf("%s",s0); insert(s0,2);
buildAC();
f[0][0][0][0]=1;//第0个点,走0步R,走0步D,取得的状态为00
for (int i=0; i<n+m; i++)//总步数
{
for (int k1=0; k1<=min(n,i); k1++)//k1步R
{
int k2=i-k1;
for (int p=0; p<=N; p++)
{
for (int k=0; k<=3; k++)
{
if (f[p][k1][k2][k])
{
if (k1<n)
{
cur=next[p][0];
num=tag[cur]|k;
f[cur][k1+1][k2][num]+=f[p][k1][k2][k];
f[cur][k1+1][k2][num]%=M;
}
if (k2<m)
{
cur=next[p][1];
num=tag[cur]|k;
f[cur][k1][k2+1][num]+=f[p][k1][k2][k];
f[cur][k1][k2+1][num]%=M;
}
}
}
}
}
}
ans=0;
for (int i=0; i<=N; i++) ans=(ans+f[i][n][m][3])%M;
printf("%d\n",ans);
}
return 0;
}

HDU 4758——Walk Through Squares——2013 ACM/ICPC Asia Regional Nanjing Online的更多相关文章

  1. hdu 4751 Divide Groups bfs (2013 ACM/ICPC Asia Regional Nanjing Online 1004)

    SDUST的训练赛 当时死磕这个水题3个小时,也无心去搞其他的 按照题意,转换成无向图,预处理去掉单向的边,然后判断剩下的图能否构成两个无向完全图(ps一个完全图也行或是一个完全图+一个孤点) 代码是 ...

  2. HDU 4749 Parade Show 2013 ACM/ICPC Asia Regional Nanjing Online

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4749 题目大意:给一个原序列N,再给出一个序列M,问从N中一共可以找出多少个长度为m的序列,序列中的数 ...

  3. [2013 ACM/ICPC Asia Regional Nanjing Online C][hdu 4750]Count The Pairs(kruskal + 二分)

    http://acm.hdu.edu.cn/showproblem.php?pid=4750 题意: 定义f(u,v)为u到v每条路径上的最大边的最小值..现在有一些询问..问f(u,v)>=t ...

  4. HDU 4751 Divide Groups 2013 ACM/ICPC Asia Regional Nanjing Online

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4751 题目大意:判断一堆人能否分成两组,组内人都互相认识. 解题思路:如果两个人不是相互认识,该两人之 ...

  5. HDU 4757 Tree(可持久化字典树)(2013 ACM/ICPC Asia Regional Nanjing Online)

    Problem Description   Zero and One are good friends who always have fun with each other. This time, ...

  6. HDU4753 Fishhead’s Little Game——2013 ACM/ICPC Asia Regional Nanjing Online

    今天比赛又是做得好水的.被狂虐啊. 比赛两个多小时一直没出题,遒遒最先交的若干发都wa了.T_T 我独自在一遍苦思了1006这个题,还好最后把这个题目A掉了,不然又是深坑队友. 题目的意思我就不多说了 ...

  7. hduoj 4710 Balls Rearrangement 2013 ACM/ICPC Asia Regional Online —— Warmup

    http://acm.hdu.edu.cn/showproblem.php?pid=4710 Balls Rearrangement Time Limit: 6000/3000 MS (Java/Ot ...

  8. hduoj 4708 Rotation Lock Puzzle 2013 ACM/ICPC Asia Regional Online —— Warmup

    http://acm.hdu.edu.cn/showproblem.php?pid=4708 Rotation Lock Puzzle Time Limit: 2000/1000 MS (Java/O ...

  9. hduoj 4715 Difference Between Primes 2013 ACM/ICPC Asia Regional Online —— Warmup

    http://acm.hdu.edu.cn/showproblem.php?pid=4715 Difference Between Primes Time Limit: 2000/1000 MS (J ...

随机推荐

  1. week14课上测试

    说明 本次测试老师将所有课下测试的易错题全部重新考察了一遍,虽然是第二次做,还提前复习过,还是错了很多,回到寝室发现老师还没有结束测试,43分的我又忍不住再做了一遍. 做第二遍发现了有几个题目是蓝墨云 ...

  2. DataGridView滚动慢​的解决方法

    当DataGridView达到一定大小的时候,拖动滚动条就会非常慢,出现让人难以忍受的闪动. 即便只有100行,每行30列. 解决方法是启用DataGridView的双缓冲. 1 2 3 4 5 6 ...

  3. python2 - 列表

    列表 a = [1,2,3,4,5,6,7] a[0:4:1]//正向索引 a[-1:-2:-1]//反向索引 列表添加 a = [1, 2] b = [3, 4] +:a + b//把a和b连接,重 ...

  4. TensorFlow深度学习实战---MNIST数字识别问题

    1.滑动平均模型: 用途:用于控制变量的更新幅度,使得模型在训练初期参数更新较快,在接近最优值处参数更新较慢,幅度较小 方式:主要通过不断更新衰减率来控制变量的更新幅度. 衰减率计算公式 : deca ...

  5. nexus实现从windows迁移至Linux平台

    说明: 由于老环境是在本地windows 2008 R2里面搭建的nexus,前面搭建了jenkins,需要将maven私库迁移至云服务器的CentOS 7系统下,之前没做过nexus的迁移,在网上看 ...

  6. 二、Django快速安装

    一.安装Python 作为一个Python Web框架,Django依赖Python.从Django适用于哪些版本的Python可以获取更多信息.较新版本的Python内置一个轻量级的数据库SQLit ...

  7. Linux 配置网络连接

    在VMware里,依次点击”编辑“ - ”虚拟网络编辑器“,如下图,我选择的是NAT模式: 在这个界面接着点"NAT设置",查看虚拟机的网关,这个网关在第三步要用.我这里的网关是1 ...

  8. 随机生成30道四则运算-NEW

    补充:紧跟上一个随机生成30道四则运算的题目,做了一点补充,可以有真分数之间的运算,于是需要在原来的基础上做一些改进. 首先指出上一个程序中的几个不足:1.每次执行的结果都一样,所以不能每天给孩子出3 ...

  9. 第二次c++作业

    用c语言实现电梯问题的方法: 先用一堆变量存储各种变量,在写一个函数模拟电梯上下移动载人放人的过程. c++: 构造一个电梯的类,用成员函数实现电梯运作的过程. 对c和c++的理解太浅,并没有感觉到用 ...

  10. jQuery异步Deferred

    原文链接:http://www.ruanyifeng.com/blog/2011/08/a_detailed_explanation_of_jquery_deferred_object.html 普通 ...