UVA - 1347 Tour

题目链接

题目大意:给你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;
}



评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值