[USACO2007FEB S]Silver Lilypad Pond
题目描述
为了让奶牛们娱乐和锻炼,农夫约翰建造了一个美丽的池塘。这个长方形的池子被分成
了M行N列个方格(1 ≤ M, N ≤ 30)。一些格子是坚固得令人惊讶的莲花,还有一些格子是
岩石,其余的只是美丽、纯净、湛蓝的水。
贝西正在练习芭蕾舞,她站在一朵莲花上,想跳到另一朵莲花上去,她只能从一朵莲花
跳到另一朵莲花上,既不能跳到水里,也不能跳到岩石上。
贝西的舞步很像象棋中的马步:每次总是先横向移动一格,再纵向移动两格,或先纵向
移动两格,再横向移动一格。最多时,贝西会有八个移动方向可供选择。
约翰一直在观看贝西的芭蕾练习,发现她有时候不能跳到终点,因为中间缺了一些荷叶。
于是他想要添加几朵莲花来帮助贝西完成任务。一贯节俭的约翰只想添加最少数量的莲花。
当然,莲花不能放在石头上。
请帮助约翰确定必须要添加的莲花的最少数量。在添加莲花最少的基础上,确定贝西从
起点跳到目标需要的最少步数。最后,确定满足添加的莲花数量最少时,步数最少的路径条
数。
输入格式
第一行:两个用空格分开的整数:M和N
第二行到M + 1行:第i + 1行有N个用空格分开的整数,描述了池塘第i行的状态:0 为水,1 为莲花,2 为岩石,3 为贝西所在的起点,4 为贝西想去的终点。
输出格式
第一行:一个整数:需要添加的莲花的最少数目;如果无解,则输出-1
第二行:一个整数:在添加莲花最少的基础上,贝西从起点跳到终点需要的最少步数;如果第一行是-1,不输出这行
第三行:一个整数:在添加莲花最少的基础上,步数等于第二行输出的路径条数;如果第一行是-1,不输出这行
样例 #1
样例输入 #1
4 8
0 0 0 1 0 0 0 0
0 0 0 0 0 2 0 1
0 0 0 0 0 4 0 0
3 0 0 0 0 0 1 0
样例输出 #1
2
6
2
首先看如何求出莲花的最小数目。我们可以这样思考,一个位置如果没有莲花,那么到他这里需要莲花,答案要加1。否则就是有莲花,不用加1.用01bfs。
然后要求最小步数。每一次都会走一步,可以直接bfs。bfs除了记录坐标,还要记录过程中添加了多少莲花。莲花数量是m*n级别的。过程中顺便再求一个路径条数,如果某个点的最少步数有多种方法到达,那么就都加上对应点的路径数。注意路径数需要开long long。
#include<bits/stdc++.h>
using namespace std;
const int dx[]={-1,-1,-2,-2,1,1,2,2},dy[]={-2,2,-1,1,-2,2,-1,1},N=35;
struct dian{
	int x,y;
};
struct node{
	int x,y,p;
};
int m,n,dis[N][N][N*N],dp[N][N],sx,sy,ex,ey,p,x,y,op[N][N],k,cx,cy,cp;
long long cnt[N][N][N*N];
deque<dian>q;
queue<node>pq;
void bfs()
{
	pq.push((node){sx,sy,0});
	dis[sx][sy][p]=0,cnt[sx][sy][p]=1;;
	while(!pq.empty())
	{
		x=pq.front().x,y=pq.front().y,p=pq.front().p;
		pq.pop();
		for(int i=0;i<8;i++)
		{
			cx=x+dx[i],cy=y+dy[i];
			if(cx>0&&cy>0&&cx<=m&&cy<=n&&op[cx][cy]!=2)
			{
				cp=p+!op[cx][cy];
				if(cp<=k)
				{
					if(!dis[cx][cy][cp])
					{
						dis[cx][cy][cp]=dis[x][y][p]+1;
						pq.push((node){cx,cy,cp});
					}
					if(dis[cx][cy][cp]==dis[x][y][p]+1)
						cnt[cx][cy][cp]+=cnt[x][y][p];
				}
			}
		}
	}
}
int main()
{
	memset(dp,0x7f,sizeof(dp));
	scanf("%d%d",&m,&n);
	for(int i=1;i<=m;i++)
	{
		for(int j=1;j<=n;j++)
		{
			scanf("%d",&op[i][j]);
			if(op[i][j]==3)
				sx=i,sy=j,op[i][j]=1;
			if(op[i][j]==4)
				ex=i,ey=j,op[i][j]=1;
		}
	}
	q.push_back((dian){sx,sy});
	dp[sx][sy]=0;
	while(!q.empty())
	{
		x=q.front().x,y=q.front().y;
		q.pop_front();
		for(int i=0;i<8;i++)
		{
			if(x+dx[i]<=m&&x+dx[i]>0&&y+dy[i]<=n&&y+dy[i]>0&&op[x+dx[i]][y+dy[i]]!=2&&dp[x+dx[i]][y+dy[i]]>dp[x][y]+!(op[x+dx[i]][y+dy[i]]))
			{
				if(!op[x+dx[i]][y+dy[i]])
				{
					dp[x+dx[i]][y+dy[i]]=dp[x][y]+1;
					q.push_back((dian){x+dx[i],y+dy[i]});
				}
				else
				{
					dp[x+dx[i]][y+dy[i]]=dp[x][y];
					q.push_front((dian){x+dx[i],y+dy[i]});
				}
			}
		}
	}
	k=dp[ex][ey];
	if(k>2e9)
	{
		printf("-1");
		return 0;
	}
	printf("%d\n",dp[ex][ey]);
	bfs();
	printf("%d\n%lld",dis[ex][ey][k],cnt[ex][ey][k]);
}
[USACO2007FEB S]Silver Lilypad Pond的更多相关文章
- 1632: [Usaco2007 Feb]Lilypad Pond
		1632: [Usaco2007 Feb]Lilypad Pond Time Limit: 5 Sec Memory Limit: 64 MBSubmit: 404 Solved: 118[Sub ... 
- BZOJ 1632: [Usaco2007 Feb]Lilypad Pond
		题目 1632: [Usaco2007 Feb]Lilypad Pond Time Limit: 5 Sec Memory Limit: 64 MBSubmit: 390 Solved: 109[ ... 
- bzoj1698 / P1606 [USACO07FEB]白银莲花池Lilypad Pond
		P1606 [USACO07FEB]白银莲花池Lilypad Pond 转化为最短路求解 放置莲花的方法如果直接算会有重复情况. 于是我们可以先预处理和已有莲花之间直接互相可达的点,将它们连边(对,忽 ... 
- 最短路【洛谷P1606】 [USACO07FEB]荷叶塘Lilypad Pond
		P1606 [USACO07FEB]荷叶塘Lilypad Pond 为了让奶牛们娱乐和锻炼,农夫约翰建造了一个美丽的池塘.这个长方形的池子被分成了M行N列个方格(1≤M,N≤30).一些格子是坚固得令 ... 
- 洛谷 P1606 [USACO07FEB]荷叶塘Lilypad Pond 解题报告
		P1606 [USACO07FEB]荷叶塘Lilypad Pond 题目描述 FJ has installed a beautiful pond for his cows' aesthetic enj ... 
- P1606 [USACO07FEB]荷叶塘Lilypad Pond(最短路计数)
		P1606 [USACO07FEB]荷叶塘Lilypad Pond 题目描述 FJ has installed a beautiful pond for his cows' aesthetic enj ... 
- [ USACO 2007 FEB ] Lilypad Pond (Silver)
		\(\\\) \(Description\) 一张\(N\times M\)的网格,已知起点和终点,其中有一些地方是落脚点,有一些地方是空地,还有一些地方是坏点. 现在要从起点到终点,每次移动走日字\ ... 
- [USACO07FEB] Lilypad Pond
		https://www.luogu.org/problem/show?pid=1606 题目描述 FJ has installed a beautiful pond for his cows' aes ... 
- poj 3271 Lilypad Pond bfs
		因为有了1的存在,使得问题变得比较难搞了,所以比较简单的做法就是把1去掉,先做一次bfs,处理出每个点能够一步到达的点(一定是1步). 然后就可以在新图上用bfs算出两个点之间的最短路,和最短路的个数 ... 
- bzoj1632 [Usaco2007 Feb]Lilypad Pond
		Description Farmer John 建造了一个美丽的池塘,用于让他的牛们审美和锻炼.这个长方形的池子被分割成了 M 行和 N 列( 1 ≤ M ≤ 30 ; 1 ≤ N ≤ 30 ) 正方 ... 
随机推荐
- Python自定义终端命令
			在python中自定义一个终端命令 这里我们想要将一个csv文件中的数据导入到数据库中,就可以定义一个终端命令,直接一行命令就可以将我们文件中的数据导入到数据库中,特别的简单 首先,我们先创建一个py ... 
- CodeForces 1343D Constant Palindrome Sum
			题意 多组样例 给一个长度为\(n\)(\(n\)一定为偶数)的数组\(a[]\),给一个正整数\(k\),保证数组内元素为小于等于\(k\)的正整数,你可以每次将数组的一个元素变为小于等于\(k\) ... 
- ThreadLocal:线程中的全局变量
			最近接了一个新需求,业务场景上需要在原有基础上新增2个字段,接口新增参数意味着很多类和方法的逻辑都需要改变,需要先判断是否属于该业务场景,再做对应的逻辑.原本的打算是在入口处新增变量,在操作数据的时候 ... 
- git status详解
			git status详解 在使用Git进行版本控制时,git status是一个非常有用的命令,用于查看当前代码仓库的状态.它可以告诉你哪些文件已更改.添加或删除,以及是否有未提交的更改等.本篇博客文 ... 
- 基于ASP.NET ZERO,开发SaaS版供应链管理系统
			前言 在园子吸收营养10多年,一直没有贡献,目前园子危机时刻,除了捐款+会员,也鼓起勇气,发篇文助力一下. 2018年下半年,公司决定开发一款SaaS版行业供应链管理系统,经过选型,确定采用ABP(A ... 
- xv6 进程切换中的锁:MIT6.s081/6.828 lectrue12:Coordination 以及 Lab6 Thread 心得
			引言 这节课和上一节xv6进程切换是一个完整的的进程切换专题,上一节主要讨论进程切换过程中的细节,而这一节主要讨论进程切换过程中锁的使用,所以本节的两大关键词就是"Coordination& ... 
- 织梦DedeCMS文章内容页调用标签方法教程
			织梦DedeCMS是新手站长使用比较多的建站软件,其模板制作也比较方便,一些站长朋友对其模板调用标签不是太熟悉,小编为大家介绍下织梦文章内容页调用标签的方法. 1.调用文章标题: {dede:fiel ... 
- 「codeforces - 1208F」Bits and Pieces
			link. 考虑把原问题写成一个在 \(\left(\log_2 \max v \right) \times n\) 的矩阵里选出三列,我们首先预处理出 \(j \cap q\).具体,我们需要对于每 ... 
- 「Semigroup と Monoid と Functional と」
			一个被国内 oi 环境弱化至几乎不存在的概念,不过我觉得还是有学一学的必要.因为我没学过代数结构所以大部分内容都在开黄腔,欲喷从轻. Semigroup 的定义是,\(\forall a,b\in\m ... 
- RK3588平台产测之ArmSoM产品低温环境测试
			1. 简介 专栏总目录 ArmSoM团队在产品量产之前都会对产品做几次专业化的功能测试以及性能压力测试,以此来保证产品的质量以及稳定性 优秀的产品都要进行严苛的多次全方位的功能测试以及性能压力测试才能 ... 
