[洛谷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 ( ...
随机推荐
- router-link-active的作用
如上图所示,创建了3个路由跳转选项,css实现后的效果如下 ↓↓↓ 当我切换“电影” “影院” “我的” 三个路由选项时,文字由黑色变成红色 此时可用vue自带的 router-link-active ...
- ASP.NET MVC 实现页落网资源分享网站+充值管理+后台管理(1)之数据库设计
本文主要讲解本项目网站所应用到的知识点,及数据库的相关设计: 一.知识点 (1)本项目主要采取ASP.NET MVC的编程模式,相信你已经了解到了MVC的具体含义是什么,这里不再赘述,有不了解的朋友, ...
- 【33.28%】【BZOJ 1195】[HNOI2006]最短母串
Time Limit: 10 Sec Memory Limit: 32 MB Submit: 1208 Solved: 402 [Submit][Status][Discuss] Descript ...
- <sUbjeCt>Reverse aAlignment SemInaR
翻译过来就是有关逆序对问题的专题. 因为大胆报名担任学校专题讲师所以跪着也要准备好课件...那什么是逆序对? 逆序对就是序列中ai>aj且i<j的有序对 举个栗子: 其中,5>4,但 ...
- C\C++串口通信编程的一点技术记录
新工作接的第一个活,要写一个配合设备调试的上位机程序. 除了MFC界面的部分,就是要处理几条命令. 串口通信部分代码借鉴的是这一篇文章:http://blog.sina.com.cn/s/blog_a ...
- 2017.5.11 昨天晚上看fview直播坚果pro回放
对话1 彭林: 我以前也是产品经理,身为产品经理,你有没有感觉我们做出过什么之前不被人接受的东西,但是我们却坚持做了,并且得到的反响非常好. 朱萧木: 没有吧,我们没有做颠覆用户认知或者三观的特别 ...
- AutoCad .Net二次开发求两曲线最小距离
测试结果: 主要思路:假设有两条曲线分别是c1和c2,把c1按照1的距离划分我这里用变量jd表示,得到一个曲线集合coll,然后遍历coll,得到coll中每一个曲线的两个端点,再用这两个端点分别求离 ...
- python列表(list)
#str #类,字符串 #name = "raitorei" #创建一个对象 #list #类,列表 ##############list类中提供的方法(灰魔法)######### ...
- C++简单实现Log日志类轻量级支持格式化输出变量
CLog 头 代码很简单 如果需要的直接Ctrl+C ----Ctrl+V 即可 #ifndef __CLOG__ #define __CLOG__ #include <windows.h&g ...
- 洛谷$P3226\ [HNOI2012]$集合选数 状压$dp$
正解:$dp$ 解题报告: 传送门$QwQ$ 考虑列一个横坐标为比值为2的等比数列,纵坐标为比值为3的等比数列的表格.发现每个数要选就等价于它的上下左右不能选. 于是就是个状压$dp$板子了$QwQ$ ...