当前位置:首页 > 题解目录 > 正文内容

【题解】马拦过河卒

亿万年的星光5年前 (2021-03-20)题解目录1801

【题目描述】

棋盘上A点有一个过河卒,需要走到目标B点。卒行走的规则:可以向下、或者向右。同时在棋盘上C点有一个对方的马,该马所在的点和所有跳跃一步可达的点称为对方马的控制点。因此称之为“马拦过河卒”。
棋盘用坐标表示,A点(0, 0)、B点(n, m)(n, m为不超过15的整数),同样马的位置坐标是需要给出的。现在要求你计算出卒从A点能够到达B点的路径的条数,假设马的位置是固定不动的,并不是卒走一步马走一步。

【输入描述】

一行四个数据,分别表示B点坐标和马的坐标。(保证所有的数据有解)

【输出描述】

一个数据,表示所有的路径条数。

【样例输入】

6 6 3 3

【样例输出】

6

【题目分析】

  • 马走日的改编版,需求遍历整个棋盘,但是这次改成了”卒“遍历棋盘(到达目的地)。

  • 卒只能向下或者向右,绝对不能走回头路,不然就可能有无数条路径了,可以用数组表示。

  • 马能走的路径至多有8个(理想情况下),可以通过数组把这8个点分别表示出来,这样给定任何一个马的坐标都可以表示其余方位。


  • 如果马在边界点上,有些坐标是不需要考虑的

  • 一定注意是”卒“躲过”马“的袭击,到达目标点,不是”马“怎么走才能吃掉”卒“

  • 题目保证有解,所以决定不存在什么”卒的起始位置就是马能跳的点“类似情况

  • DFS深搜加上判断条件即可,注意m和n的最大是15

  • 本题使用框架二,用flag数组表示马能走的点。

  • int Search(int k)
     {
       if  (到目的地) 输出解;
       else
        for (i=1;i<=算符种数;i++)
         if  (满足条件) 
           {
            保存结果;
                         Search(k+1);
            恢复:保存结果之前的状态{回溯一步}
           }
     }


   


【参考答案】


#include<iostream>
#include<cstdio>
#include<cstring>
using namespace std;
int x,y,hx,hy; //定义B点坐标和马的坐标
int dx[2]= {0,1}; //卒的x变化
int dy[2]= {1,0}; //卒的y变化
bool flag[16][16]; //标记数组,用来标记哪些点能走或不能走
int sum; //定义总路径条数
void changeFlag(int x,int y) { //马能走的所有路径都封死,不让卒走
	flag[x][y]=1;
	flag[x-1][y-2]=1;
	flag[x-2][y-1]=1;
	flag[x-2][y+1]=1;
	flag[x-1][y+2]=1;
	flag[x+1][y-2]=1;
	flag[x+2][y-1]=1;
	flag[x+2][y+1]=1;
	flag[x+1][y+2]=1;
	//此处有个小问题,就是如果马的坐标在左边界的时候,数组有可能出现flag[-1][0]这类越界问题。但是,题目保证有解,所以在这个地方没有额外考虑
}
//使用框架二
void dfs(int sx,int sy) {
	if(sx==x && sy==y) {
		sum++;  //到达b点就累加1
		return;
	}
	for(int i=0; i<2; i++) { //卒只有两走法(下,右)
		int fx=sx+dx[i]; //当前卒的某一方向的x坐标
		int fy=sy+dy[i]; //当前卒的某一方向的y坐标
		if(flag[fx][fy]==0 && fx<=x && fy<=y) { 
		//如果当前这个点没有在马的目标点上,而且没有超过目标点x,y 
			flag[fx][fy]=1; //将当前点变为1,表示此点已经走过了 
			dfs(fx,fy); //继续遍历下一个点 
			flag[fx][fy]=0; //回溯一步 
		}
	}

}
int main() {
	cin>>x>>y>>hx>>hy;
	changeFlag(hx,hy); //把马走的路径封死
	dfs(0,0); //从0,0开始遍历,注意,是 卒 的遍历,不是马
	cout<<sum<<endl;
	return 0;
}



扫描二维码推送至手机访问。

版权声明:本文由青少年编程知识记录发布,如需转载请注明出处。

分享给朋友:

相关文章

2021年市北区程序设计竞赛题(⼩学组)

最⼤值的相乘(maxx.cpp)【问题描述】第⼀⾏有x个正整数a1,a2,..,ax,第⼆⾏有y个正整数b1,b2,...,by,第三⾏有z个正整数c1,c2,...,cz,假设第⼀⾏的x个正整数中的...

【题解】单词接龙

【题目描述】单词接龙是一个与我们经常玩的成语接龙相类似的游戏,现在我们已知一组单词,且给定一个开头的字母,要求出以这个字母开头的最长的“龙”(每个单词都最多在“龙”中出现两次),在两个单词相连时,其重...

【题解】移动路线

【题目描述】X桌子上有一个m行n列的方格矩阵,将每个方格用坐标表示,行坐标从下到上依次递增,列坐标从左至右依次递增,左下角方格的坐标为(1,1),则右上角方格的坐标为(m,n)。小明是个调皮的孩子,一...

猴子吃桃

【题目描述】猴子吃桃问题。猴子第一天摘下若干个桃子,当即吃了一半,还不过瘾,又多吃了一个。 第二天早上又将剩下的桃子吃掉一半,又多吃一个。以后每天早上都吃了前一天剩下的一半零一个。到第N天早上想再吃时...

【题解】约瑟夫问题2

【题解】约瑟夫问题2

【题目描述】M个人围成一圈,每分钟相邻的两个人可以交换位置(只能有一对交换)。求使M个人的顺序颠倒(即每个人左边相邻的人换到右边,右边相邻的人换到左边)所需的最少时间(分钟数)。【输入描述】 ...

糖果传递

题目描述有n个小朋友坐成一圈,每人有ai个糖果。每人只能给左右两人传递糖果。每人每次传递一个糖果代价为1。输入格式第一行一个正整数n≤1000000,表示小朋友的个数.接下来n行,每行一个整数ai,表...