uva1347 Tour
题目大意:
两个人同时从最左点出发,沿着两条不同的路径走到最右点(除了起点和终点每个点走且仅走一次)
/* 状态 dp[i][j]指当前两人分别走到i,j点。且设i>j; 则有:dp[i+1][i]=min (dp[i+1][i],dp[i][j]+dist[i][i+1]); dp[i+1][j]=min (dp[i+1][j],dp[i][j]+dist[j][i+1]); 第二个走到i+1时本应转移到dp[i][i+1],但是根据此处的规定,必须写成dp[i+1][i] 因为每个点都要走到,所以每次走到没走到的第一个点。 最终结果是 dp[n-1][i]+dist[i][n]+dist[n-1][n];(0<i<n-1) 最终点要走两次。 */ #include<iostream> #include<cstdio> #include<cstring> #include<cmath> using namespace std; int n; double dp[1010][1010],dis[1010][1010]; struct node{ int x,y; }point[1010]; double count(int a,int b){ double x1=point[a].x,x2=point[b].x,y1=point[a].y,y2=point[b].y; return sqrt((x1-x2)*(x1-x2)+(y1-y2)*(y1-y2)); } int main(){ while(scanf("%d",&n)!=EOF){ for(int i=1;i<=n;i++) scanf("%d%d",&point[i].x,&point[i].y); for(int i=1;i<=n;i++) for(int j=1;j<=n;j++){ if(i!=j)dis[i][j]=count(i,j); dp[i][j]=0x3fffffff; } dp[1][1]=0; for(int i=1;i<n;i++) for(int j=1;j<n;j++){ dp[i+1][j]=min(dp[i+1][j],dp[i][j]+dis[i][i+1]); dp[i+1][i]=min(dp[i+1][i],dp[i][j]+dis[i+1][j]); } double ans=dp[n][n-1]+dis[n][n-1]; printf("%.2lf\n",ans); } }