#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;
}