题目大意:给你n个不相同的点,按照x坐标的不同从左向右排列,希望你能给出一个方案,从左边的点走到最右边 的点,然后从最右边的点再回来,中间出了开头和结尾的点经过了两次,其他的点都只经过一次,输出最短的这个路程距离
题目分析:披着几何外皮的记忆化搜索问题,只是这个递推方程我们很难想,我看了解析也想了半天这个递推方程
解释:dp(i, j)表示(1~max(i, j))全部都走过,并且前面的 i 始终大于 j ,为了防止有漏的点,我们让站在i, j 两个位置上的两个人,每次只能有其中一个移动到 i + 1 这个位置,这样就出现了我们的状态转移方程, 每一次可以是 i -> i+1 这样子就是 dp(i+1, j ) + dist(i, i+1), 也可以是 j -> i+1这样子算的结果就是 dp(i+1, i) + diat(j, i+1),因为是要求最短距离,两者之中选最小就可以,这样就得出了我们的状态转移方程
Code:
#include<iostream>
#include<cstdio>
#include<cstring>
#include<cmath>
using namespace std;
const int maxn = 1010;
double dp[maxn][maxn];
struct PPoint{
double x, y;
// int x, y;
};
typedef PPoint Point;
Point P[maxn];
int n;
double dist(int i, int j)
{
double dx = P[i].x - P[j].x;
double dy = P[i].y - P[j].y;
// return sqrt(dx*dx + dy*dy);
return hypot(dx, dy); //???什么操作,这也太神奇了
//计算三角形的斜边长
}
double DP(int i, int j)
{
double & ans = dp[i][j];
if(ans > 0) return ans;
if(i == n-1)
return ans = dist(i, n) + dist(j, n);
ans = min(DP(i+1, j)+dist(i, i+1), DP(i+1, i)+dist(j, i+1));
return ans;
}
int main()
{
while(scanf("%d", &n) != EOF){
memset(dp, 0, sizeof(dp));
for(int i = 1; i <= n; i++)
scanf("%lf %lf", &P[i].x, &P[i].y);
DP(2, 1);
printf("%.2lf\n", dist(1,2)+dp[2][1]);
}
return 0;
}