UVA1347 Tour

2021.5.22
刷题的时候突然看到手机推送,袁隆平院士逝世,心中一颤,后来得到辟谣,心情稍微放松几分,正在刷着辟谣的文章时,央视新闻发文,13点07分,袁隆平院士逝世,没过多久又看到吴孟超院士逝世的新闻,心情难以平复,特在本文的开头,向两位院士致敬。
历史浩荡,国士无双。

UVA1347 Tour

题目链接

dp题,按照紫书上的分析做下来的,下面主要也是跟着紫书走一遍。

题目分析

“从左到右再回来”不太方便思考,可以改成:两个人同时从最左点出发,沿着两条不同的路径走,最后都走到最右点,且除了起点和终点外其余每个点恰好被一个人经过。这样,就可以用d(i,j)表示一个人走到i,另一个人走到j,还需要走的距离。
但是,如此定义,很难保证两个人不会走到同一点。
下面修改一下,d(i,j)表示从1~max(i,j)全部走过,且当前两人一个在i一个在j还需要走的距离,并且规定下一步只能走到i+1,即下一个状态为d(i+1,j)或d(i+1,i)(相当于从j点到i+1)我们规定把大的序号放在前面,否则可能会出现死循环。
边界条件是d(n-1,j)=dist(n-1,n)+dist(j,n),dist表示两点间的距离
状态转移方程为d(i,j) = min(d(i + 1, j) + dist(i, i + 1), d(i + 1, i) + dist(j, i + 1))

AC代码

#include<iostream>
#include<cmath>
#include<iomanip>
#include<cstring>
#include<algorithm>
using namespace std;
const int MAXN = 1000 + 10;
struct point
{
	int x, y;
	point(int a = 0, int b = 0)
	{
		x = a;
		y = b;
	}
};
point points[MAXN];
double distance(int i, int j)
{
	point a = points[i];
	point b = points[j];
	double dis = sqrt(pow((a.x - b.x), 2) + pow((a.y - b.y), 2));
	return dis;
}
int n;
double dp[MAXN][MAXN];
double d(int i, int j)
{
	if (dp[i][j] > 0)
		return dp[i][j];
	if (i == n - 1)
		return dp[i][j] = distance(i, n) + distance(j, n);
	return dp[i][j] = min(d(i + 1, j) + distance(i, i + 1), d(i + 1, i) + distance(j, i + 1));
}
int main()
{
	while (cin >> n)
	{
		memset(dp, 0, sizeof(dp));
		for (int i = 1; i <= n; i++)
		{
			cin >> points[i].x >> points[i].y;
		}
		cout << fixed << setprecision(2) << d(2, 1) + distance(1, 2) << endl;
	}
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值