[ USACO 2007 FEB ] Lilypad Pond (Silver)

本文深入探讨了一种在N×M网格中寻找从起点到终点的最短路径算法,特别关注通过开发空地来减少步数和路径数量的优化策略。文章详细介绍了如何通过构建图模型,使用SPFA算法解决这一问题,同时提供了代码实现细节。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

\(\\\)

\(Description\)


一张\(N\times M\)的网格,已知起点和终点,其中有一些地方是落脚点,有一些地方是空地,还有一些地方是坏点。

现在要从起点到终点,每次移动走日字\((\)横一纵二或横二纵一\()\),其中只能经过起点、终点、落脚点。

现在可以开发任意个数的空地变为落脚点,问找到合法路径最少需要开发多少个空地,在满足第一个条件下最少移动多少步,在满足前两个条件下有多少条不同的路径。

  • \(N,M\in [1,30]\)

\(\\\)

\(Solution\)


被上一个题干蒙直接\(NC\)......我还说Silver咋比Gold还难

这题.......仔细读题之后错觉是\(Gold\)那题再加上一层限制的最短路,后来发现不是.......

仔细读题,注意这次的方案数不再是放置落脚点的方案,而是路径数。

那么我们就不必考虑选择不同原有落脚点导致方案同构的尴尬情况了,直接将每一个点向一步可以到达的点建边,如果是从空地向外建边代价为\(1\),其他为\(0\)即可。

最短路计数时有双层限制,注意讨论更新的情况。

\(\\\)

\(Code\)


#include<cmath>
#include<queue>
#include<cctype>
#include<cstdio>
#include<cstring>
#include<cstdlib>
#include<iostream>
#include<algorithm>
#define N 50
#define M 60010
#define R register
#define gc getchar
#define inf 9000000000000000ll
using namespace std;
typedef long long ll;
 
inline int rd(){
   int x=0; bool f=0; char c=gc();
   while(!isdigit(c)){if(c=='-')f=1;c=gc();}
   while(isdigit(c)){x=(x<<1)+(x<<3)+(c^48);c=gc();}
   return f?-x:x;
}
 
bool vis[N*N];
 
const int dx[8]={1,1,-1,-1,2,2,-2,-2};
 
const int dy[8]={2,-2,2,-2,1,-1,1,-1};
 
ll ts[N*N],stp[N*N],dis[N*N];
int n,m,s,t,tot,cnt,num[N][N],mp[N][N],hd[N*N];
 
struct edge{int w,to,nxt;}e[M<<1];
 
inline void add(int u,int v,int w){
  e[++tot].to=v; e[tot].w=w;
  e[tot].nxt=hd[u]; hd[u]=tot;
}
 
inline void make(int ux,int uy){
  for(R int i=0,nx,ny;i<8;++i){
    nx=ux+dx[i]; ny=uy+dy[i];
    if(nx<1||nx>n||ny<1||ny>m) continue;
    add(num[ux][uy],num[nx][ny],mp[ux][uy]==0);
  }
}
 
queue<int> q;
 
inline void SPFA(){
  memset(vis,0,sizeof(vis));
  memset(stp,0x3f,sizeof(stp));
  for(R int i=1;i<=cnt;++i) dis[i]=inf;
  q.push(s); dis[s]=0;
  ts[s]=1ll; stp[s]=0;
  while(!q.empty()){
    int u=q.front();
    q.pop(); vis[u]=0;
    for(R int i=hd[u],v;i;i=e[i].nxt)
      if(dis[v=e[i].to]>dis[u]+e[i].w){
        dis[v]=dis[u]+e[i].w;
        stp[v]=stp[u]+1; ts[v]=ts[u];
        if(!vis[v]) vis[v]=1,q.push(v);
      }
      else if(dis[v]==dis[u]+e[i].w&&stp[v]>stp[u]+1){
            stp[v]=stp[u]+1; ts[v]=ts[u];
            if(!vis[v]) vis[v]=1,q.push(v);
          }
      else if(dis[v]==dis[u]+e[i].w&&stp[v]==stp[u]+1) ts[v]+=ts[u];
  }
}
 
int main(){
  n=rd(); m=rd();
  for(R int i=1;i<=n;++i)
    for(R int j=1;j<=m;++j){
      mp[i][j]=rd();
      num[i][j]=++cnt;
      if(mp[i][j]==3) s=cnt;
      if(mp[i][j]==4) t=cnt;
    }
  for(R int i=1;i<=n;++i)
    for(R int j=1;j<=m;++j)
      if(mp[i][j]!=2) make(i,j);
  SPFA();
  if(dis[t]<inf) printf("%lld\n%lld\n%lld\n",dis[t],stp[t],ts[t]);
  else puts("-1");
  return 0;
}

转载于:https://www.cnblogs.com/SGCollin/p/9737730.html

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值