【BZOJ1560】【JSOI2009】火星藏宝图 动规

#include <stdio.h>
int main()
{
	puts("转载请注明出处谢谢");
	puts("http://blog.csdn.net/vmurder/article/details/43029513");
}


题解:

我们先把点排序,优先排行。

呃,我们或许可以暴力建边(但如果a->b,b->c,则a不连c)

但是如果把正方形拆成四个,然后左上角往右上来10W个,然后……

等会,我想错了,这种方法貌似可以做,自己脑补一下去吧,拓扑图DP。


下面说正解:

我们扫一遍这些点,

记录pos[i]表示第i列[当前]最靠下的有点位置,然后f[i]记录它的动规值。

然后O(M)扫一遍得到当前点的动规值,更新pos和f。


然后最后f[m]就是答案,别忘了在最开始把(1,1)和(m,m)这俩岛加进去。


WA了两遍的原因是sort虽然是稳定的排序,但是架不住你自己重载,

这样开始加进去的(1,1)可能不是第一个,而我是从第二个开始扫的,导致某岛1的水果就被popoqqq吃了。

其实不用加岛一的。。o(︶︿︶)o 、、


代码:

#include <cstdio>
#include <cstring>
#include <iostream>
#include <algorithm>
#define N 1050
#define M 201000
#define INF 0x3f3f3f3f3f3f3f3fll
using namespace std;
struct YYC
{
	int x,y,p;
	bool operator < (const YYC &a)const{return x==a.x?y<a.y:x<a.x;}
	void read(){scanf("%d%d%d",&x,&y,&p);}
}a[M];
int pos[N];
long long f[N];
int n,m;
int main()
{
	freopen("test.in","r",stdin);
	int i,j,k;
	scanf("%d%d",&n,&m),n+=2;
	a[1].x=a[1].y=1;
	a[2].x=a[2].y=m;
	for(i=3;i<=n;i++)a[i].read();
	sort(a+2,a+n+1);

	pos[1]=1;
	for(i=2;i<=n;i++)
	{
		long long temp=-INF;
		for(j=1;j<=a[i].y;j++)if(pos[j])
			temp=max(temp,f[j]-(a[i].y-j)*(a[i].y-j)-(a[i].x-pos[j])*(a[i].x-pos[j]));
		pos[a[i].y]=a[i].x,f[a[i].y]=temp+a[i].p;
	}
	printf("%d\n",f[m]);
	return 0;
}


你可能感兴趣的:(动规,BZOJ1560,JSOI2009,火星藏宝图)