Problem

给你2*n的格子,每个格子有一个字母,从任意一点出发,不重复的经过上下左右,生成要求的字符串。问有几种不同的走法。

Solution



分三段,左U型、中间、右U型。

分别枚举左边和右边的长度,中间一段用Dp来解决。

Dp[i][j][k],i,j,k表示当前在(i,j)位置,枚举到第k个字符。

Notice

特殊情况下有重复。

Code

#include<cmath>
#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
using namespace std;
#define sqz main
#define ll long long
#define reg register int
#define rep(i, a, b) for (reg i = a; i <= b; i++)
#define per(i, a, b) for (reg i = a; i >= b; i--)
#define travel(i, u) for (reg i = head[u]; i; i = edge[i].next)
const int INF = 1e9, mod = 1e9 + 7, Ha = 826036489, N = 2000;
const double eps = 1e-6, phi = acos(-1.0);
ll modd(ll a, ll b) {if (a >= b || a < 0) a %= b; if (a < 0) a += b; return a;}
ll read(){ ll x = 0; int zf = 1; char ch; while (ch != '-' && (ch < '0' || ch > '9')) ch = getchar();
if (ch == '-') zf = -1, ch = getchar(); while (ch >= '0' && ch <= '9') x = x * 10 + ch - '0', ch = getchar(); return x * zf;}
void write(ll y) { if (y < 0) putchar('-'), y = -y; if (y > 9) write(y / 10); putchar(y % 10 + '0');}
char S[2][N + 5], st[N + 5];
ll f[2][N + 5][N + 5], mi[N + 5];
int n, m;
void Calc(ll &X, ll Y)
{
X += Y;
if (X >= mod) X -= mod;
}
struct Hash
{
ll hash[N + 5];
void Make(int n, char *s)
{
hash[0] = 0;
rep(i, 1, n) hash[i] = (hash[i - 1] * 31 + s[i] - 'a') % Ha;
}
ll Cut(int l, int r)
{
return (hash[r] - hash[l - 1] * mi[r - l + 1] % Ha + Ha) % Ha;
}
}pre[2], suf[2], Comp;
ll Solve(int flag)
{
ll T = 0;
memset(f, 0, sizeof(f));
rep(j, 1, n)
{
f[0][j][0] = f[1][j][0] = 1;
rep(i, 0, 1)
{
rep(k, 2, min(n - j + 1, m / 2))
if (Comp.Cut(m - 2 * k + 1, m - k) == pre[i].Cut(j, j + k - 1) && Comp.Cut(m - k + 1, m) == suf[1 - i].Cut(n - (j + k - 1) + 1, n - j + 1))
if (2 * k != m || flag) Calc(T, f[i][j][m - 2 * k]);
rep(k, 2, min(j, m / 2))
if (Comp.Cut(k + 1, 2 * k) == pre[i].Cut(j - k + 1, j) && Comp.Cut(1, k) == suf[1 - i].Cut(n - j + 1, n - (j - k + 1) + 1))
if (2 * k != m || flag) Calc(f[i][j + 1][2 * k], 1);
}
rep(i, 0, 1)
rep(k, 0, m - 1)
if (S[i][j] == st[k + 1])
{
Calc(f[i][j + 1][k + 1], f[i][j][k]);
if (k + 2 <= m && S[1 - i][j] == st[k + 2])
Calc(f[1 - i][j + 1][k + 2], f[i][j][k]);
}
rep(i, 0, 1) Calc(T, f[i][j + 1][m]);
}
return T;
} int sqz()
{
scanf("%s%s%s", S[0] + 1, S[1] + 1, st + 1);
n = strlen(S[0] + 1), m = strlen(st + 1);
mi[0] = 1;
rep(i, 1, 2000) mi[i] = (mi[i - 1] * 31) % Ha;
rep(i, 0, 1)
{
pre[i].Make(n, S[i]);
reverse(S[i] + 1, S[i] + n + 1);
suf[i].Make(n, S[i]);
reverse(S[i] + 1, S[i] + n + 1);
}
Comp.Make(m, st);
if (m == 1)
{
printf("%I64d\n", Solve(1) % mod);
return 0;
}
ll ans = 0;
Calc(ans, Solve(1));
reverse(st + 1, st + m + 1);
Comp.Make(m, st);
Calc(ans, Solve(0));
if (m == 2)
rep(i, 1, n)
{
if (S[0][i] == st[1] && S[1][i] == st[2]) Calc(ans, mod - 1);
if (S[1][i] == st[1] && S[0][i] == st[2]) Calc(ans, mod - 1);
}
printf("%I64d\n", ans);
return 0;
}

[Codeforces613E]Puzzle Lover的更多相关文章

  1. CF613E Puzzle Lover

    题意 英文版题面 Problems Submit Status Standings Custom test .input-output-copier { font-size: 1.2rem; floa ...

  2. cf 613E - Puzzle Lover

    Description 一个\(2*n\)的方格矩阵,每个格子里有一个字符 给定一个长度为\(m\)的字符串\(s\) 求在方格矩阵中,有多少种走法能走出字符串\(s\) 一种合法的走法定义为:从任意 ...

  3. 题解 CF613E Puzzle Lover

    解题思路 其实仔细观察我们可以发现路径一定是一个类似于下图的一个左括号之后中间随便反复曲折,然后右边在来一个右括号. 然后对于两个括号形状的东西其实是可以利用 Hash 来判等特殊处理的. 对于中间的 ...

  4. codeforces613E

    Puzzle Lover CodeForces - 613E Oleg Petrov loves crossword puzzles and every Thursday he buys his fa ...

  5. cf Round 613

    A.Peter and Snow Blower(计算几何) 给定一个点和一个多边形,求出这个多边形绕这个点旋转一圈后形成的面积.保证这个点不在多边形内. 画个图能明白 这个图形是一个圆环,那么就是这个 ...

  6. 多校联训 DP 专题

    [UR #20]跳蚤电话 将加边变为加点,方案数为 \((n-1)!\) 除以一个数,\(dp\) 每种方案要除的数之和即可. 点击查看代码 #include<bits/stdc++.h> ...

  7. Puzzle 面向服务/切面(AOP/IOC)开发框架 For .Net

    Puzzle 面向服务/切面AOP开发框架 For .Net AOP主要实现的目的是针对业务处理过程中的切面进行提取,它所面对的是处理过程中的某个步骤或阶段,以获得逻辑过程中各部分之间低耦合性的隔离效 ...

  8. HDU5456 Matches Puzzle Game(DP)

    题目 Source http://acm.hdu.edu.cn/showproblem.php?pid=5456 Description As an exciting puzzle game for ...

  9. one recursive approach for 3, hdu 1016 (with an improved version) , permutations, N-Queens puzzle 分类: hdoj 2015-07-19 16:49 86人阅读 评论(0) 收藏

    one recursive approach to solve hdu 1016, list all permutations, solve N-Queens puzzle. reference: t ...

随机推荐

  1. qt 安装包生成

    (Qt Installer Framework)程序简易打包教程 2017年06月19日 14:38:47 carman_风 阅读数:3559 标签: installerqt框架 更多 个人分类: 软 ...

  2. db2空值、null

    1. 输入参数为字符类型,且允许为空的,可以使用COALESCE(inputParameter,'')把NULL转换成''; 2. 输入类型为整型,且允许为空的,可以使用COALESCE(inputP ...

  3. OpenStack-Neutron-VPNaaS-配置

    配置openstack版本:Juno vpnaas配置的资料很少,官网目前参考的https://wiki.openstack.org/wiki/Neutron/VPNaaS/HowToInstall比 ...

  4. element table 二次封装 父子组件传值 组件通信

    新建一个组件(即子组件)table.vue 子组件编辑内容如下图所示 子组件通过props获取父组件传递过来的参数,如下图所示,type指明传递到子组件的数据类型,default指定默认值,一般不给 ...

  5. 论文笔记【二】Making Sense of Word Embeddings

    摘要 1.作者提出了一种新的简单有效的方法,用于学习词义嵌入word sense embedding 2.传统的两种方法:(1)直接从语料库中学习词义:(2)依赖词汇资源的语义库 研究方法的创新点:通 ...

  6. java面试题汇总(有的题无视即可,没什么实际用途)

    相关概念 面向对象的三个特征 封装,继承,多态,这个应该是人人皆知,有时候也会加上抽象. 多态的好处 允许不同类对象对同一消息做出响应,即同一消息可以根据发送对象的不同而采用多种不同的行为方式(发送消 ...

  7. Linux 文件时间记录属性 调优

    Linux 文件时间属性介绍 atime:(access time)显示的是文件中的数据最后被访问的时间,比如系统的进程直接使用或通过一些命令和脚本间接使用.(执行一些可执行文件或脚本) mtime: ...

  8. kubeflow 创建tensorflow过程

    online deployable ,install k8s 代码 Kubeflow有三个核心组件 TFJob Operator 和 Controller: 作为Kubernetes的扩展,来简化分布 ...

  9. SSM-网站后台管理系统制作(1)

    好久没写博客了,忙于考试和项目答辩,今天整理一下想弄的SSM:本人想做的是博客管理平台,和博客园,CSDN,stackoverflow这些类似. 老师先让做的是后台管理系统,先给出来吧. (讲解内容: ...

  10. onclick 事件

    onclick 事件 Event 对象 定义和用法 onclick 事件会在对象被点击时发生. 请注意, onclick 与 onmousedown 不同.单击事件是在同一元素上发生了鼠标按下事件之后 ...