[洛谷P1606] [USACO07FEB] 荷叶塘Lilypad Pond
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的更多相关文章
- 洛谷 P1606 [USACO07FEB]荷叶塘Lilypad Pond 解题报告
P1606 [USACO07FEB]荷叶塘Lilypad Pond 题目描述 FJ has installed a beautiful pond for his cows' aesthetic enj ...
- 洛谷 P1606 [USACO07FEB]荷叶塘Lilypad Pond【spfa】
和bzoj同名题不一样! 起点和水点向花费一个荷花能到的第一个点连一条边权为1的有向边,然后跑计数spfa即可 #include<iostream> #include<cstdio& ...
- P1606 [USACO07FEB]荷叶塘Lilypad Pond(最短路计数)
P1606 [USACO07FEB]荷叶塘Lilypad Pond 题目描述 FJ has installed a beautiful pond for his cows' aesthetic enj ...
- 【luogu P1606 [USACO07FEB]荷叶塘Lilypad Pond】 题解
题目链接:https://www.luogu.org/problemnew/show/P1606 这个题..第一问很好想,但是第二问,如果要跑最短路计数的话,零边权的花怎么办? 不如这样想,如果这个点 ...
- 最短路【洛谷P1606】 [USACO07FEB]荷叶塘Lilypad Pond
P1606 [USACO07FEB]荷叶塘Lilypad Pond 为了让奶牛们娱乐和锻炼,农夫约翰建造了一个美丽的池塘.这个长方形的池子被分成了M行N列个方格(1≤M,N≤30).一些格子是坚固得令 ...
- Dfs+Spfa【p1606】[USACO07FEB]荷叶塘Lilypad Pond
Description 为了让奶牛们娱乐和锻炼,农夫约翰建造了一个美丽的池塘.这个长方形的池子被分成了M行N列个方格(1≤M,N≤30).一些格子是坚固得令人惊讶的莲花,还有一些格子是岩石,其余的只是 ...
- LuoguP1606 [USACO07FEB]荷叶塘Lilypad Pond 【最短路】By cellur925
最短路好题!] 参考资料:学长 https://blog.csdn.net/TSOI_Vergil/article/details/52975779 学长太强了!!!%%% 题目传送门 ======= ...
- 洛谷——P1821 [USACO07FEB]银牛派对Silver Cow Party
P1821 [USACO07FEB]银牛派对Silver Cow Party 题目描述 One cow from each of N farms (1 ≤ N ≤ 1000) conveniently ...
- 洛谷P2875 [USACO07FEB]牛的词汇The Cow Lexicon
P2875 [USACO07FEB]牛的词汇The Cow Lexicon 题目描述 Few know that the cows have their own dictionary with W ( ...
随机推荐
- linux 从用户空间的 I/O 存取
刚刚描述的这些函数主要打算被设备驱动使用, 但它们也可从用户空间使用, 至少在 PC- 类 的计算机. GNU C 库在 <sys/io.h> 中定义它们. 下列条件应当应用来对于 inb ...
- ASP.NET MVC 实现页落网资源分享网站+充值管理+后台管理(1)之数据库设计
本文主要讲解本项目网站所应用到的知识点,及数据库的相关设计: 一.知识点 (1)本项目主要采取ASP.NET MVC的编程模式,相信你已经了解到了MVC的具体含义是什么,这里不再赘述,有不了解的朋友, ...
- Linux 内核kobject 层次, kset, 和子系统
kobject 结构常常用来连接对象到一个层级的结构中, 匹配正被建模的子系统的结构. 有 2 个分开的机制对于这个连接: parent 指针和 ksets. 在结构 kobject 中的 paren ...
- Linux USB 和 sysfs
由于单个 USB 物理设备的复杂性, 设备在 sysfs 中的表示也非常复杂. 物理 USB 设备 (通过 struct usb_device 表示)和单个 USB 接口(由 struct usb_i ...
- 2019 ICPC 陕西西安邀请赛 D. Miku and Generals
传送门:https://nanti.jisuanke.com/t/39271 题意: 给你n个人,每个人有一个权值 a_i ,(a_i是可以被100整除的))现在需要你将n个人分成两组,有m个关系 ...
- Java 学习笔记(7)——接口与多态
上一篇说了Java面向对象中的继承关系,在继承中说到:调用对象中的成员变量时,根据引用类型来决定调用谁,而调用成员方法时由于多态的存在,具体调用谁的方法需要根据new出来的对象决定,这篇主要描述的是J ...
- SpringBoot如何优雅的使用RocketMQ
目录 SpringBoot如何优雅的使用RocketMQ SpringBoot如何优雅的使用RocketMQ MQ,是一种跨进程的通信机制,用于上下游传递消息.在传统的互联网架构中通常使用MQ来对上下 ...
- 第二阶段:4.商业需求文档MRD:6.PRD-其他需求
- 洛谷$1156$ 垃圾陷阱 $dp$
\(Sol\) \(f_{i,j}\)前\(i\)个垃圾,能活到时间\(j\)的最高垃圾高度.\(t_i\)表示第\(i\)个垃圾掉落的时间,\(g_i\)表示吃垃圾\(i\)能维持的时间,\(h_i ...
- acwing 1250. 格子游戏 并查集
地址 https://www.acwing.com/problem/content/1252/ Alice和Bob玩了一个古老的游戏:首先画一个 n×nn×n 的点阵(下图 n=3n=3 ). 接着, ...