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