hdu 4568(SPFA预处理+TSP)

题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4568

思路:先用spfa预处理出宝藏与宝藏之间的最短距离,宝藏到边界的最短距离,然后就是经典的求TSP过程了。

  1 #include<iostream>

  2 #include<cstdio>

  3 #include<cstring>

  4 #include<algorithm>

  5 #include<queue>

  6 using namespace std;

  7 #define MAXN 222

  8 #define inf 1<<30

  9 

 10 struct Point{

 11     int x,y;

 12 }point[MAXN];

 13 

 14 int value[MAXN][MAXN];//原图

 15 int map[MAXN][MAXN];//宝藏间的最短距离

 16 int dist[MAXN][MAXN];//宝藏到边界的距离

 17 int dd[MAXN];//宝藏到达边界的最短距离

 18 bool mark[MAXN][MAXN];

 19 int dp[1<<14][14];

 20 int n,m,k;

 21 int dir[4][2]={{-1,0},{1,0},{0,-1},{0,1}};

 22 

 23 void spfa(int num)

 24 {

 25     memset(mark,false,sizeof(mark));

 26     for(int i=0;i<n;i++)

 27         for(int j=0;j<m;j++)dist[i][j]=inf;

 28     queue<pair<int,int> >que;

 29     que.push(make_pair(point[num].x,point[num].y));

 30     if(dist[point[num].x][point[num].y]==-1)return;

 31     dist[point[num].x][point[num].y]=0;

 32     while(!que.empty()){

 33         int x=que.front().first,y=que.front().second;

 34         que.pop();

 35         mark[x][y]=false;

 36         if(x==0||x==(n-1)||y==0||y==(m-1)){

 37             dd[num]=min(dd[num],dist[x][y]);

 38         }

 39         for(int i=0;i<4;i++){

 40             int xx=x+dir[i][0],yy=y+dir[i][1];

 41             if(xx>=0&&xx<n&&yy>=0&&yy<m&&value[xx][yy]!=-1){

 42                 if(dist[x][y]+value[xx][yy]<dist[xx][yy]){

 43                     dist[xx][yy]=dist[x][y]+value[xx][yy];

 44                     if(!mark[xx][yy]){

 45                         mark[xx][yy]=true;

 46                         que.push(make_pair(xx,yy));

 47                     }

 48                 }

 49             }

 50         }

 51     }

 52 }

 53 

 54 

 55 int main()

 56 {

 57     int _case;

 58     scanf("%d",&_case);

 59     while(_case--){

 60         scanf("%d%d",&n,&m);

 61         for(int i=0;i<n;i++)

 62             for(int j=0;j<m;j++)

 63                 scanf("%d",&value[i][j]);

 64         scanf("%d",&k);

 65         for(int i=0;i<k;i++)scanf("%d%d",&point[i].x,&point[i].y);

 66         for(int i=0;i<k;i++)

 67             for(int j=0;j<k;j++)

 68                 map[i][j]=(i==j)?0:inf;

 69         for(int i=0;i<(1<<k);i++)

 70             for(int j=0;j<k;j++)dp[i][j]=inf;

 71         fill(dd,dd+MAXN,inf);

 72         for(int i=0;i<k;i++){

 73             spfa(i);

 74             for(int j=0;j<k;j++){

 75                 if(i==j)continue;

 76                 map[i][j]=min(map[i][j],dist[point[j].x][point[j].y]);//宝藏与宝藏之间的最近距离

 77             }

 78             dp[1<<i][i]=dd[i]+value[point[i].x][point[i].y];//宝藏到边的最近距离

 79         }

 80         for(int s=0;s<(1<<k);s++){

 81             for(int i=0;i<k;i++){

 82                 if(s&(1<<i)==0)continue;

 83                 if(dp[s][i]==inf)continue;

 84                 for(int j=0;j<k;j++){

 85                     if(s&(1<<j)==1)continue;

 86                     dp[s|(1<<j)][j]=min(dp[s|(1<<j)][j],dp[s][i]+map[i][j]);

 87                 }

 88             }

 89         }

 90         int ans=inf;

 91         for(int i=0;i<k;i++){

 92             ans=min(ans,dp[(1<<k)-1][i]+dd[i]);//拿到了所有的宝藏之后还要走出来

 93         }

 94         printf("%d\n",ans);

 95     }

 96     return 0;

 97 }

 98 

 99 

100             

101 

102 

103         
View Code

 

你可能感兴趣的:(SPFA)