题目:蓝桥杯之过河马
资源限制
时间限制:1.0s 内存限制:256.0MB
问题描述
在那个过河卒逃过了马的控制以超级超级多的走法走到了终点之后,这匹马表示它不开心了……
于是,终于有一天,它也过河了!
由于过河马积累了许多的怨念,所以这次它过了河之后,再也没有什么东西可以限制它,它可以自由自在的在棋盘上驰骋。一开始,它是在一个n行m列棋盘的左下角(1,1)的位置,它想要走到终点右上角(n,m)的位置。而众所周知,马是要走日子格的。可是这匹马在积累了这么多怨念之后,它再也不想走回头路——也就是说,它只会朝向上的方向跳,不会朝向下的方向跳。
那么,这匹马它也想知道,它想从起点跳到终点,一共有多少种走法呢?
输入格式
第一行两个数n,m,表示一个n行m列的棋盘,马最初是在左下角(1,1)的位置,终点在右上角(n,m)的位置。
输出格式
输出有一行,一个数表示走法数。由于答案可能很大,所以输出答案除以1000000007所得的余数即可。
样例输入
4 4
样例输出
2
数据规模和约定
n<=100,m<=100
题解:
这题的关键就是需要整理清楚每个坐标中的走法种类是怎么出现的,是累加呢?还是累乘呢?
带着这个疑问,我们进入棋盘中寻找答案。
**1.**可以看到,由于本题中规定了只能往上走,不能往下走。以红圈中的点为例,红圈所处的点只能由绿圈标记的以下4个位置得到,因此将这四个点的走法累加至红圈处即可。
亦可初步推得,每个点的位置都是由这样的相似情况累加得到。
由此,该题中走法的种类是由递推累加得到的。
即c[i][j]= c[i - 1][j - 2]+ c[i - 2][j - 1]+c[i -2][j + 1]+c[i -1 ][j +2 ];
(此处尚未考虑大小越界问题)
**2.**其次,我们还需考虑一点——绿圈的位置必须在棋盘之内
因此,我们在加起来之前,还需对其是否在理应存在的范围
(左>=1,下>=1,右<=m)之内做出判定
代码如下
#include<iostream>
#include<cstring>
#include<string>
#include<cstdio>
using namespace std;
int main()
{
long long c[102][102]={0};
c[1][1] = 0;
c[2][3] = 1;
c[3][2] = 1;
//在坐标3,2和2,3处的走法只能由1,1直接过来取得,故可先对其初始化
int n, m;
cin >> n >> m;
for(int i =1;i<=n;i++)
for (int j = 1; j <= m; j++)
{
if(i-1>=1&&j-2>=1)
c[i][j] += c[i - 1][j - 2];
//最开始用int c[][]而不是longlong 的话,每一部都需要取余
if (i - 2 >= 1 && j - 1 >= 1)
c[i][j] += c[i - 2][j - 1];
if (i - 2 >= 1 && j + 1 <=m)
c[i][j] += c[i -2][j + 1];
if (i - 1 >= 1 && j + 2 <= m)
c[i][j] += c[i -1 ][j +2 ];
c[i][j]%= 1000000007;
}
cout << c[n][m];
return 0;
}