蓝桥杯之过河马

题目:蓝桥杯之过河马

资源限制
时间限制: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;
}

  • 4
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值