Description###

为了让奶牛们娱乐和锻炼,农夫约翰建造了一个美丽的池塘。这个长方形的池子被分成了M行N列个方格(1≤M,N≤30)。一些格子是坚固得令人惊讶的莲花,还有一些格子是岩石,其余的只是美丽、纯净、湛蓝的水。

贝西正在练习芭蕾舞,她站在一朵莲花上,想跳到另一朵莲花上去,她只能从一朵莲花跳到另一朵莲花上,既不能跳到水里,也不能跳到岩石上。

贝西的舞步很像象棋中的马步:每次总是先横向移动一格,再纵向移动两格,或先纵向移动两格,再横向移动一格。最多时,贝西会有八个移动方向可供选择。

约翰一直在观看贝西的芭蕾练习,发现她有时候不能跳到终点,因为中间缺了一些荷叶。于是他想要添加几朵莲花来帮助贝西完成任务。一贯节俭的约翰只想添加最少数量的莲花。当然,莲花不能放在石头上。

请帮助约翰确定必须要添加的莲花的最少数量,以及有多少种放置这些莲花的方法.

Input###

第一行:两个用空格分开的整数:M和N

第二行到M+1行:第i+1行有N个用空格分开的整数,描述了池塘第i行的状态:

0为水,1为莲花,2为岩石,3为贝西所在的起点,4为贝西想去的终点。

Output###

第一行:一个整数,需要增加的最少莲花数;如果无解,输出-1。

第二行:放置这些莲花的方案数量,保证这个数字不会超过一个64位的有符号整数,

如果第一行是-1,不要输出第二行。

Sample Input###

4 5

1 0 0 0 0

3 0 0 0 0

0 0 2 0 0

0 0 0 4 0

Sample Output###

2

3


想法##

这个题还是挺好玩的。

看到这个题后我的第一反应是dp,就是记f[i]为这个点到终点要添加的最少莲花数,转移就枚举所有i能一次跳到的点j,若该点为水则f[i]=f[j]+1,否则f[i]=f[j]

然后发现这样其实就是建了一张图,每个点向一次能跳到的点连边,若它跳到的那个点为水则边权为1,否则边权为0

这样跑一个最短路就是需要添加的最少莲花数

然后考虑怎么计数。

首先需要意识到,题目要求的是放莲花的方案,而不是上图最短路的数量

那么我们可以尝试把所有0边都删掉,在剩下的边权都为1的图中最短路数量便是放莲花方案数

怎么删0边呢,其实就是把通过0边连起来的两个点直接连一条边就行了

换句话说,就是每一个点向它在添加一次莲花后可跳到的点连边,原本有莲花的点就不放在新图中了

跑个最短路计数就好了


代码##

#include<cstdio>
#include<iostream>
#include<algorithm>
#include<queue> #define INF 100000000 using namespace std; typedef long long ll;
const int N = 40;
const int M = N*N; struct node{
int v;
node *next;
}pool[M*1000],*h[M];
int cnt;
void addedge(int u,int v){
node *p=&pool[++cnt];
p->v=v;p->next=h[u];h[u]=p;
} queue<int> que;
ll d[M],num[M];
int vis[M];
int S,T,n,m; ll get(int u){
if(num[u]) return num[u];
int v;
for(node *p=h[u];p;p=p->next)
if(d[v=p->v]==d[u]-1) num[u]+=get(v);
return num[u];
}
void spfa(){
int u,v;
while(!que.empty()) que.pop();
for(int i=1;i<=n*m;i++) d[i]=INF;
que.push(S); vis[S]=1; d[S]=0;
while(!que.empty()){
u=que.front(); que.pop();
for(node *p=h[u];p;p=p->next)
if(d[v=p->v]>d[u]+1){
d[v]=d[u]+1;
if(!vis[v]) { que.push(v); vis[v]=1; }
}
vis[u]=0;
}
num[S]=1;
for(int i=1;i<=n*m;i++)
if(!num[i]) get(i);
} int map[N][N],dre[8][2]={-2,-1,-2,1,-1,-2,-1,2,1,-2,1,2,2,-1,2,1}; int used[N][N];
bool check(int x,int y){ return x>0 && x<=n && y>0 && y<=m; }
void add(int id,int x,int y){
int xx,yy;
used[x][y]=id;
for(int i=0;i<8;i++)
if(check(xx=x+dre[i][0],yy=y+dre[i][1])){
if(used[xx][yy]==id || map[xx][yy]==2) continue;
used[xx][yy]=id;
if(map[xx][yy]==1) add(id,xx,yy);
else addedge(id,(xx-1)*m+yy);
}
} int main()
{
scanf("%d%d",&n,&m);
for(int i=1;i<=n;i++)
for(int j=1;j<=m;j++) {
scanf("%d",&map[i][j]);
if(map[i][j]==3) S=(i-1)*m+j;
else if(map[i][j]==4) T=(i-1)*m+j;
}
for(int i=1;i<=n;i++)
for(int j=1;j<=m;j++) if(map[i][j]!=1 && map[i][j]!=2) add((i-1)*m+j,i,j); spfa();
if(d[T]==INF) printf("-1\n");
else printf("%lld\n%lld\n",d[T]-1,num[T]); return 0;
}

[洛谷P1606] [USACO07FEB] 荷叶塘Lilypad Pond的更多相关文章

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

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

  2. 洛谷 P1606 [USACO07FEB]荷叶塘Lilypad Pond【spfa】

    和bzoj同名题不一样! 起点和水点向花费一个荷花能到的第一个点连一条边权为1的有向边,然后跑计数spfa即可 #include<iostream> #include<cstdio& ...

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

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

  4. 【luogu P1606 [USACO07FEB]荷叶塘Lilypad Pond】 题解

    题目链接:https://www.luogu.org/problemnew/show/P1606 这个题..第一问很好想,但是第二问,如果要跑最短路计数的话,零边权的花怎么办? 不如这样想,如果这个点 ...

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

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

  6. Dfs+Spfa【p1606】[USACO07FEB]荷叶塘Lilypad Pond

    Description 为了让奶牛们娱乐和锻炼,农夫约翰建造了一个美丽的池塘.这个长方形的池子被分成了M行N列个方格(1≤M,N≤30).一些格子是坚固得令人惊讶的莲花,还有一些格子是岩石,其余的只是 ...

  7. LuoguP1606 [USACO07FEB]荷叶塘Lilypad Pond 【最短路】By cellur925

    最短路好题!] 参考资料:学长 https://blog.csdn.net/TSOI_Vergil/article/details/52975779 学长太强了!!!%%% 题目传送门 ======= ...

  8. 洛谷——P1821 [USACO07FEB]银牛派对Silver Cow Party

    P1821 [USACO07FEB]银牛派对Silver Cow Party 题目描述 One cow from each of N farms (1 ≤ N ≤ 1000) conveniently ...

  9. 洛谷P2875 [USACO07FEB]牛的词汇The Cow Lexicon

    P2875 [USACO07FEB]牛的词汇The Cow Lexicon 题目描述 Few know that the cows have their own dictionary with W ( ...

随机推荐

  1. Error与Exception的区别,Java常见异常Execption总结

    错误和异常的区别(Error vs Exception) 错误和异常的区别(Error vs Exception) 今天面试问了这样一个问题,"Error" 和 "Exc ...

  2. Kobjects, Ksets 和 Subsystems

    Kobject 是基础的结构, 它保持设备模型在一起. 初始地它被作为一个简单的引用计数, 但是它的责任已随时间增长, 并且因此有了它自己的战场. struct kobject 所处理的任 务和它的支 ...

  3. es6笔记 day4---模块化

    模块化: 注意:需要放到服务器环境 1.如何定义模块? export  东西 export   const a = 12; export { a  as  aaa, b  as  banana } 2 ...

  4. javax.el.PropertyNotFoundException: Property 'XXX' not found on type java.lang.String

    遇到的问题: 在使用idea开发Java Web时,调用SSM框架出现了如下错误: 但是我的类中已经定义了geter和seter方法,如下: 而Jsp中的调用代码是通过EL实现,也导入了相应的包.如下 ...

  5. dotnet Framework 源代码 · Ink

    本文是分析 .NET Framework 源代码的系列,主要告诉大家微软做笔迹用的思路,怎么做的笔迹才是高性能的,用户体验比较好的.我会告诉大家源代码的思想,当然这个文章会比较无聊.如果你是想做笔迹的 ...

  6. Curator源码阅读 - ConnectionState的管理与监听

    看看Curator框架 为实现对 连接状态ConnectionState的监听,都是怎么构造框架的.后面我们也可以应用到业务的各种监听中. Curator2.13实现 接口 Listener List ...

  7. attr(),prop()二者区别和最佳实践

    attr(),prop()二者区别和最佳实践 最近使用到attr()来获取自定义属性值,我印象中是有一个方法可以获取到自定义属性值,进而我又想到了另一个方法prop().  查看了手册发现并没有对二者 ...

  8. 闲着没事,做个chrome浏览器插件,适合初学者

    时光偷走的,永远都是我们眼皮底下看不见的珍贵. 本插件功能:替换掉网页中的指定图片的src地址. 使用插件前: 使用插件后: 鲜花(闲话):这个网站的不加水印的图片连接被保存在,图片的data-ima ...

  9. Java面向对象程序设计第7章1-8

    Java面向对象程序设计第7章1-8 1."程序中凡是可能出现异常的地方必须进行捕获或拋出",这句话对吗? 不对. 异常分两类,runtime异常和非runtime异常. runt ...

  10. Hadoop集群分布式安装

    一 整体介绍 1.1 硬件环境 本文使用三台服务器搭建hadoop集群,使用Centos7.5系统,服务器均有独立ip 1.2 部署的软件 部署服务:namenode(HA),resourcemana ...