题目描述

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

了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. SpringBoot3.x原生镜像-Native Image实践

    前提 之前曾经写过一篇<SpringBoot3.x 原生镜像-Native Image 尝鲜>,当时SpringBoot处于3.0.0-M5版本,功能尚未稳定.这次会基于SpringBoo ...

  2. CF939F Cutlet 题解

    题意简述 有一个正反面都为 \(0\) 的卡片,每过 \(1\) 分朝下那一面的数值就会增加 \(1\),你可以在几个区间的时间内翻转卡片,求经过 \(2n\) 秒后能否让这个卡片的正反面的数都为 \ ...

  3. MIT6.s081/6.828 lectrue07:Page faults 以及 Lab5 心得

    本篇博客主要是复习 MIT6.s081/6.828 lectrue07:Page faults 以及记录 Lab5 :COW fork 的心得 值得一提的是,2020 年之前的版本第 5 个 lab ...

  4. 全局多项式(趋势面)与IDW逆距离加权插值:MATLAB代码

      本文介绍基于MATLAB实现全局多项式插值法与逆距离加权法的空间插值的方法,并对不同插值方法结果加以对比分析. 目录 1 背景知识 2 实际操作部分 2.1 空间数据读取 2.2 异常数据剔除 2 ...

  5. 聊一聊使用Spring事物时不生效的场景

    前言 今天介绍一下Spring事物不生效的场景,事物是我们在项目中经常使用的,如果是Java的话,基本上都使用Spring的事物,不过Spring的事物如果使用不当,那么就会导致事物失效或者不回滚,最 ...

  6. chatglm2-6b在P40上做LORA微调

    背景: 目前,大模型的技术应用已经遍地开花.最快的应用方式无非是利用自有垂直领域的数据进行模型微调.chatglm2-6b在国内开源的大模型上,效果比较突出.本文章分享的内容是用chatglm2-6b ...

  7. Wood,微型 Java ORM 框架(首次发版)

    Wood,微型 Java ORM 框架(支持:java sql,xml sql,annotation sql:事务:缓存:监控:等...),零依赖! 特点和理念: 跨平台:可以嵌入到JVM脚本引擎(j ...

  8. 6.1 KMP算法搜索机器码

    KMP算法是一种高效的字符串匹配算法,它的核心思想是利用已经匹配成功的子串前缀的信息,避免重复匹配,从而达到提高匹配效率的目的.KMP算法的核心是构建模式串的前缀数组Next,Next数组的意义是:当 ...

  9. 5 分钟理解 Next.js SSG (Static Site Generation / Static Export)

    5 分钟理解 Next.js SSG (Static Site Generation / Static Export) 在本篇文章中,我们将介绍 Next.js 中的 SSG(静态网站生成)功能,以及 ...

  10. c语言代码练习15

    //使用togo跳转代码,阻止关机#define _CRT_SECURE_NO_WARNINGS 1 #include <stdio.h> #include <string.h> ...