一只猪走进了一个森林。很凑巧的是,这个森林的形状是长方形的,有n行,m列组成。我们把这个长方形的行从上到下标记为1到n,列从左到右标记为1到m。处于第r行第c列的格子用(r,c)表示。
刚开始的时候猪站在(1,1),他的目标是走到(n,m)。由于猪回家心切,他在(r,c)的时候,只会往(r+1,c)或(r,c+1)走。他不能走出这个森林。
这只猪所在的森林是一个非同寻常的森林。有一些格子看起来非常相似,而有一些相差非常巨大。猪在行走的过程中喜欢拍下他经过的每一个格子的照片。一条路径被认为是漂亮的当且仅当拍下来的照片序列顺着看和反着看是一样的。也就是说,猪经过的路径要构成一个回文。
数一数从(1,1)到(n,m)有多少条漂亮路径。答案可能非常巨大,请输出对 109+7 取余后的结果。
样例解释:有三种可能
Input
单组测试数据。
第一行有两个整数 n,m (1≤n,m≤500),表示森林的长和宽。
接下来有n行,每行有m个小写字母,表示每一个格子的类型。同一种类型用同一个字母表示,不同的类型用不同的字母表示。
Output
输出答案占一行。
Input示例
3 4
aaab
baaa
abba
Output示例
3 这就是道Dp题 和之前1的传纸条有点像 但这个是一个从(1,1)出发 一个从(n,m)出发
f【i】【j】【k】表示一个点到(i,j)另一个到(k,l)
l可以从另外三个数推出来 因为两个点所走的路程一定一样
#include<cstdio>
#include<cstring>
#include<algorithm>
#define LL long long
using namespace std;
const int mod=1e9+;
const int M=;
int read(){
int ans=,f=,c=getchar();
while(c<''||c>''){if(c=='-') f=-; c=getchar();}
while(c>=''&&c<=''){ans=ans*+(c-''); c=getchar();}
return ans*f;
}
char s[M][M];
int f[][M][M],now=,last=;
int h,n,m,ans;
int main()
{
n=read(); m=read(); h=(n+m)>>;
for(int i=;i<=n;i++) scanf("%s",s[i]+);
if(s[][]!=s[n][m]){printf("0\n"); return ;}
f[][][n]=;
for(int i=;i<=n;i++){
for(int j=;j<=(h-i+);j++){
for(int k=n;k>=max(,n-i-j+);k--){
int l=n+m-i-j-k+;
if(s[i][j]!=s[k][l]) continue;
(f[now][j][k]+=f[last][j][k])%=mod;
(f[now][j][k]+=f[last][j][k+])%=mod;
(f[now][j][k]+=f[now][j-][k+])%=mod;
(f[now][j][k]+=f[now][j-][k])%=mod;
if((i==k&&j==l)||(i==k&&j+==l)||(i+==k&&j==l)) (ans+=f[now][j][k])%=mod;
}
}
memset(f[last],,sizeof(f[last]));
swap(last,now);
}printf("%d\n",ans);
return ;
}