一傻逼题调了两天。。

n<=30 * m<=30的地图,0表示可以放平台,1表示本来有平台,2表示不能走,3起点4终点,走路方式为象棋的日字,求:从起点走到终点,至少要放多少平台,以及放平台的方案数,无解-1。

方法一:其实能走直接平台的就可以直接走来走去,也就是算一个联通块。类似于tarjan,先把一大块缩成一点,然后连边走最短路。

错误!存在边权为0的边,会导致统计方案出现重复。比如:

圆圈走到三角形,直接走和绕一圈是一样的,但算了两次。

方法二:把0边去掉就行了。由于数据小,开个数组[a][b][c][d]表示a,b到c,d是否能通过填一块到达,这个数组只需对每个点做一次搜索就能出来。最后就是最短路啦。

不过边权为1,谁最短路会写迪杰呢?肯定bfs啦!

 #include<stdio.h>
#include<string.h>
#include<stdlib.h>
#include<algorithm>
//#include<iostream>
using namespace std; int n,m;
int a[][],can[][][][];
#define maxn 1011
#define maxm 10011
struct Edge{int tx,ty,next;}edge[maxn*maxn];int first[][],le=;
#define LL long long
void in(int sx,int sy,int tx,int ty)
{
Edge &e=edge[le];
e.tx=tx;e.ty=ty;
e.next=first[sx][sy];
first[sx][sy]=le++;
}
int nx,ny;
const int dx[]={,,,,-,-,-,-},dy[]={,-,,-,,-,,-};
bool vis[][];
void dfs(int x,int y)
{
vis[x][y]=;
for (int i=;i<;i++)
{
const int xx=x+dx[i],yy=y+dy[i];
if (xx< || xx>n || yy< || yy>m || can[nx][ny][xx][yy]
|| a[xx][yy]== || vis[xx][yy]) continue;
if (a[xx][yy]== || a[xx][yy]==)
{
can[nx][ny][xx][yy]=;
continue;
}
dfs(xx,yy);
}
}
const int inf=0x3f3f3f3f;
int dis[][];LL way[][];
struct qnode{int x,y;}q[maxn];int head,tail;
void bfs(int sx,int sy)
{
head=;tail=;q[].x=sx;q[].y=sy;
for (int i=;i<=n;i++)
for (int j=;j<=m;j++)
dis[i][j]=inf;
dis[sx][sy]=;way[sx][sy]=;
while (head!=tail)
{
const int nx=q[head].x,ny=q[head++].y;
for (int i=first[nx][ny];i;i=edge[i].next)
{
const Edge &e=edge[i];
if (dis[e.tx][e.ty]>dis[nx][ny]+)
{
dis[e.tx][e.ty]=dis[nx][ny]+;
way[e.tx][e.ty]=way[nx][ny];
q[tail].x=e.tx,q[tail++].y=e.ty;
}
else if (dis[e.tx][e.ty]==dis[nx][ny]+)
way[e.tx][e.ty]+=way[nx][ny];
}
}
}
int main()
{
scanf("%d%d",&n,&m);
int sx,sy,tx,ty;
for (int i=;i<=n;i++)
for (int j=;j<=m;j++)
{
scanf("%d",&a[i][j]);
if (a[i][j]==) sx=i,sy=j;
if (a[i][j]==) tx=i,ty=j;
}
for (int i=;i<=n;i++)
for (int j=;j<=m;j++)
{
memset(can[i][j],,sizeof(can[i][j]));
memset(vis,,sizeof(vis));
if (a[i][j]==) continue;
nx=i;ny=j;
dfs(i,j);
}
memset(first,,sizeof(first));
for (int i=;i<=n;i++)
for (int j=;j<=m;j++)
for (int k=;k<=n;k++)
for (int l=;l<=m;l++)
{
if (can[i][j][k][l])
{
in(i,j,k,l);
}
}
bfs(sx,sy);
if (dis[tx][ty]==inf) puts("-1");
else printf("%d\n%lld\n",dis[tx][ty]-,way[tx][ty]);
return ;
}

BZOJ1698: [Usaco2007 Feb]Lilypad Pond 荷叶池塘的更多相关文章

  1. BZOJ 1632: [Usaco2007 Feb]Lilypad Pond

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

  2. 1632: [Usaco2007 Feb]Lilypad Pond

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

  3. 「BZOJ 1698」「USACO 2007 Feb」Lilypad Pond 荷叶池塘「最短路」

    题解 从一个点P可以跳到另一个点Q,如果Q是水这条边就是1,如果Q是荷叶这条边权值是0.可以跑最短路并计数 问题是边权为0的最短路计数没有意义(只是荷叶的跳法不同),所以我们两个能通过荷叶间接连通的点 ...

  4. bzoj1632 [Usaco2007 Feb]Lilypad Pond

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

  5. BZOJ1632: [Usaco2007 Feb]Lilypad Pond SPFA+最短路计数

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

  6. BZOJ 1632 [Usaco2007 Feb]Lilypad Pond:spfa【同时更新:经过边的数量最小】【路径数量】

    题目链接:http://www.lydsy.com/JudgeOnline/problem.php?id=1632 题意: 有一个n*m的池塘.0代表水,1代表荷花,2代表岩石,3代表起点,4代表终点 ...

  7. 【BZOJ】1632: [Usaco2007 Feb]Lilypad Pond(bfs)

    http://www.lydsy.com/JudgeOnline/problem.php?id=1632 我简直是个sb... ... bfs都不会写.. 算方案还用2个bfs! 都不会整合到一个! ...

  8. bzoj 1632: [Usaco2007 Feb]Lilypad Pond【bfs】

    直接bfs,在过程中更新方案数即可 #include<iostream> #include<cstdio> #include<queue> using namesp ...

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

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

随机推荐

  1. knockout Observable Array(监控数组)

    Observable Array(监控数组)的作用 列表操作是经常会遇到的一个场景,使用监控数组,你可以: 保存列表对象,并且使用Ko提供的丰富的API操作列表元素(支持内建js Array的方法,以 ...

  2. js递归和数组去重(简单便捷的用法)

    1.递归例子<script type="text/javascript"> function test(num) { if(num < 0) { return; ...

  3. ubuntu系统apache日志文件的位置

    Debian,Ubuntu或Linux Mint上的Apache错误日志位置 默认的错误日志 在基于Debian的Linux上,系统范围的Apache错误日志默认位置是/var/log/apache2 ...

  4. Halcon学习笔记2

    图像预处理 一般来说,我们采集到的图像会有一些小黑点,小斑点,不平滑等因素会会影响我们后期的算法,此时就需要我们对其图片进行预处理. 下面是一些预处理基本算子: 1.消除噪声:mean_image/b ...

  5. 原创:四种Linux系统开机启动项优命令超给力超详细详解

    老葵花哥哥又开课了 接下来是你们的齐天大圣孙悟空给你们带来的详细版Linux系统开机启动优化四种命令 第一种方法是很正常的 第二种有点难理解 第三种来自我的一个奇思妙想 本文档秉承 不要钱也不要臀部的 ...

  6. select2宽度占比100%,导致无法实现浮动效果

  7. mysql数据库比较,各数据库不同之处

    和mysql数据库比较,各数据库不同之处: Oracle数据库:字段类型不同 postgresql数据库:show tables不同; SQL语句需要前面加上 模式名 Mongodb数据库:文档存储, ...

  8. ssh 带密码私钥 输入密码

    $ssh-agent bash $ssh-add -k ~/.ssh/id_rsa Enter passphrase for /home/ubuntu/.ssh/id_rsa: Identity ad ...

  9. c++_方格分割

    标题:方格分割 6x6的方格,沿着格子的边线剪开成两部分.要求这两部分的形状完全相同. 如图:p1.png, p2.png, p3.png 就是可行的分割法. 试计算:包括这3种分法在内,一共有多少种 ...

  10. OAuth2.0授权流程

    微信授权 第3步的微信授权中的scope字段: snsapi_base 静默授权,不弹出用户同意框,可直接获取成员的基础信息:    snsapi_userinfo:静默授权,弹出用户同意框,待用户同 ...