码迷,mamicode.com
首页 > 其他好文 > 详细

[UVA 12589]Learning Vector[DP]

时间:2015-08-12 23:41:19      阅读:157      评论:0      收藏:0      [点我收藏+]

标签:uva   dp   

题目链接:[UVA 12589]Learning Vector[DP]
题意分析:给出n个矢量,从中选择k个,以坐标原点为起点,收尾相连,问:这样的k个周围相连矢量与x轴围成图形的最大面积的两倍是多少?

解题思路:考虑状态:dp[id][pick][h]代表到第id个矢量为止,选择pick个矢量离最大面积还差多少,h为当前图形最右端高度。具体转移看代码。

这里着重说一下为什么要对这些矢量按斜率进行排序:

首先,整个求解过程其实就是在暴力枚举这些向量的组合,只不过采用了记忆化搜索优化。

其次,对于选出来的矢量,我们一定按照斜率大的先放的策略进行放置。理由如下:

对于任意选定的一组矢量收尾相加(这里的矢量都满足x>=0,y>=0),其最终最右端高度是一样的,其水平宽度是一样的。每一个矢量都为它的后继矢量提供了一个基底高度,下图红色部分即为第一个矢量提供的基底高度,斜率高的矢量具有高度高,长度短的特点,为后继提供的面积更大。至此,粗略证明了贪心的正确性。

技术分享

个人感受:嘛,大部分时间都花在了理解贪心上。总的来说这题还是蛮有收获的,想过用状压枚举子集,然后发现50个点哭瞎,没想到还能这样枚举~
具体代码如下:
#include<iostream>
#include<cctype>
#include<cstdio>
#include<cstring>
#include<string>
#include<cmath>
#include<algorithm>
using namespace std;
typedef long long ll;
const int INF = 0x3f3f3f3f;
const int MAXN = 55;
int vis[MAXN][MAXN][3000], dp[MAXN][MAXN][3000], t, n, k, kase;
struct V{
	int x, y;
	bool operator <(const V &t)const
	{
		return y * t.x > x * t.y; //斜率[y/x]>[t.y/t.x]
	}
}v[MAXN];

int dfs(int id, int pick, int h)
{
	int &ret = dp[id][pick][h];
	if (pick == k) return ret = 0; //到这里,离最大面积只剩0了
	if (id == n) return ret = -INF; //不满足条件,返回-INF
	if (vis[id][pick][h] == kase) return ret;
	vis[id][pick][h] = kase;
	ret = 0;
	ret = max(ret, dfs(id + 1, pick, h)); //不取第个id个
	ret = max(ret, dfs(id + 1, pick + 1, h + v[id].y) + 2*v[id].x*h+v[id].x*v[id].y); //取第id个
	return ret;
}

int main()
{
	for (kase = scanf("%d",&t); kase <= t; ++kase)
	{
	    cin >> n >> k;
	    for (int i = 0; i < n; ++i) cin >> v[i].x >> v[i].y;
	    sort(v, v + n);
	    cout << "Case " << kase << ": " << dfs(0,0,0) << '\n';
	}
	return 0;
}

版权声明:欢迎转载(^ω^)~不过转载请注明原文出处:http://blog.csdn.net/catglory ?(????)

[UVA 12589]Learning Vector[DP]

标签:uva   dp   

原文地址:http://blog.csdn.net/catglory/article/details/47451547

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!