题目描述

为了让奶牛们娱乐和锻炼,农夫约翰建造了一个美丽的池塘。这个长方形的池子被分成

了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的更多相关文章

  1. 1632: [Usaco2007 Feb]Lilypad Pond

    1632: [Usaco2007 Feb]Lilypad Pond Time Limit: 5 Sec  Memory Limit: 64 MBSubmit: 404  Solved: 118[Sub ...

  2. BZOJ 1632: [Usaco2007 Feb]Lilypad Pond

    题目 1632: [Usaco2007 Feb]Lilypad Pond Time Limit: 5 Sec  Memory Limit: 64 MBSubmit: 390  Solved: 109[ ...

  3. bzoj1698 / P1606 [USACO07FEB]白银莲花池Lilypad Pond

    P1606 [USACO07FEB]白银莲花池Lilypad Pond 转化为最短路求解 放置莲花的方法如果直接算会有重复情况. 于是我们可以先预处理和已有莲花之间直接互相可达的点,将它们连边(对,忽 ...

  4. 最短路【洛谷P1606】 [USACO07FEB]荷叶塘Lilypad Pond

    P1606 [USACO07FEB]荷叶塘Lilypad Pond 为了让奶牛们娱乐和锻炼,农夫约翰建造了一个美丽的池塘.这个长方形的池子被分成了M行N列个方格(1≤M,N≤30).一些格子是坚固得令 ...

  5. 洛谷 P1606 [USACO07FEB]荷叶塘Lilypad Pond 解题报告

    P1606 [USACO07FEB]荷叶塘Lilypad Pond 题目描述 FJ has installed a beautiful pond for his cows' aesthetic enj ...

  6. P1606 [USACO07FEB]荷叶塘Lilypad Pond(最短路计数)

    P1606 [USACO07FEB]荷叶塘Lilypad Pond 题目描述 FJ has installed a beautiful pond for his cows' aesthetic enj ...

  7. [ USACO 2007 FEB ] Lilypad Pond (Silver)

    \(\\\) \(Description\) 一张\(N\times M\)的网格,已知起点和终点,其中有一些地方是落脚点,有一些地方是空地,还有一些地方是坏点. 现在要从起点到终点,每次移动走日字\ ...

  8. [USACO07FEB] Lilypad Pond

    https://www.luogu.org/problem/show?pid=1606 题目描述 FJ has installed a beautiful pond for his cows' aes ...

  9. poj 3271 Lilypad Pond bfs

    因为有了1的存在,使得问题变得比较难搞了,所以比较简单的做法就是把1去掉,先做一次bfs,处理出每个点能够一步到达的点(一定是1步). 然后就可以在新图上用bfs算出两个点之间的最短路,和最短路的个数 ...

  10. bzoj1632 [Usaco2007 Feb]Lilypad Pond

    Description Farmer John 建造了一个美丽的池塘,用于让他的牛们审美和锻炼.这个长方形的池子被分割成了 M 行和 N 列( 1 ≤ M ≤ 30 ; 1 ≤ N ≤ 30 ) 正方 ...

随机推荐

  1. 1、MyBatis简介

    1.1.MyBatis历史 MyBatis最初是Apache的一个开源项目iBatis, 2010年6月这个项目由Apache Software Foundation迁 移到了Google Code. ...

  2. C++ ASIO 实现异步套接字管理

    Boost ASIO(Asynchronous I/O)是一个用于异步I/O操作的C++库,该框架提供了一种方便的方式来处理网络通信.多线程编程和异步操作.特别适用于网络应用程序的开发,从基本的网络通 ...

  3. 试试用Markdown来设计表单

    相信很多后端开发.对于前端知识是比较零碎的,所以很多时候写表单这样的工作,一般就是复制黏贴,然后改改字段.对于HTML格式,一直觉得比较杂乱,不够简洁. 最近TJ发现了一个有趣的小工具:Create ...

  4. 后端常用的Linux命令大全,建议收藏

    引言 作为一名后端工程师,使用终端是一种常见的做法,也是你应该学习的技能.许多命令和实用程序可以帮助你在使用 Linux 时更有效地完成任务. 基本 Linux 命令 如果你想使用 Linux 操作系 ...

  5. 使用 Sealos 一键部署高可用 MinIO,开启对象存储之旅

    大家好!今天这篇文章主要向大家介绍如何通过 Sealos 一键部署高可用 MinIO 集群. MinIO 对象存储是什么? 对象是二进制数据,例如图像.音频文件.电子表格甚至二进制可执行代码.对象的大 ...

  6. Solution Set -「CF 1539」

    我是傻逼. 「CF 1539A」Contest Start Link. 答案是 \(\sum_{i=1}^{n-1}\min\{i,\lfloor\frac{t}{x}\rfloor\}\),等差数列 ...

  7. Redis系列之——Redis介绍安装配置

    文章目录 第一章 redis初识 1.1 Redis是什么 1.2 Redis特性(8个) 1.3 Redis单机安装 1.3.1下载安装 1.3.2三种启动方式 1.3.2.1 最简启动 1.3.2 ...

  8. OpenSSL 生成 RootCA (根证书)并自签署证书(支持 IP 地址)

    背景 某机房内部访问需要配置 HTTPS,网上找的一些证书教程都不是特别好,有些直接生成证书,没有根 CA 的证书导致信任不了 Ubuntu 机器,有些教程只有域名生成,没有 IP 生成,有些甚至报错 ...

  9. CF1343C

    题目简化和分析: 给您一个序列,您要在其中选择若干个数使得: 相邻两数异号 长度最大,总和最大 我们可以牢牢抓住长度且总和最大,这一特性. 说明我们必须在每一个连续的同号的子串中被迫选择最大的,以满足 ...

  10. Android下音视频对讲演示程序(声学回音消除、噪音抑制、语音活动检测、自动增益控制、自适应抖动缓冲)(2023年07月13日更新)

    Android下音视频对讲演示程序 必读说明 简介   本软件根据<道德经>为核心思想而设计,实现了两个设备之间进行音视频对讲,一般可用于楼宇对讲.智能门铃对讲.企业员工对讲.智能对讲机. ...