【bzoj 1588】 [HNOI2002]营业额统计 双向链表

参考论文:http://wenku.baidu.com/link?url=qTuT3ghZfvaygqB4a15bL8jN-gNt9BV-A_zOv3HvpSEvR3Ujoz4VbsZwjc7gISRVgoz9QNE6ZNLP1D0T3muLxKh_x5hgMfu57TJd4UwB5va

其他应用:noip 2012 开车旅行 题解:http://blog.csdn.net/pbihao/article/details/52875298

其实也是在黄学长那里看到的qaq

先把输入数据从小到大排序,然后建一个数组记录下原来的序列中的每一个元素对应新的序列中的数的序号,把新的序列记录下左右连边的编号(双向链表),这样的话,我们从n到1处理原数列的每一个元素,min(a[j]-a[i])就一定在i的左边或者右边,每一次处理完以后就删除链表中的这一个元素就可以保证所有的元素(左右)都在这个元素之前出现过了

#include
#include
#include
#include
#include
#include
#define maxn 40200
using namespace std;
int n,a[maxn],b[maxn];
struct C{
	int x,id,ls,rs;
	bool operator<(const C& b)const {return x1;i--){
		int nw=b[i],ll=c[b[i]].ls,rr=c[b[i]].rs;
		if(ll>=1&&rr<=n){
			ans+=min(abs(c[ll].x-c[nw].x),abs(c[rr].x-c[nw].x));
			c[ll].rs=rr,c[rr].ls=ll;
		}else if(ll>=1){
			ans+=abs(c[ll].x-c[nw].x);
			c[ll].rs=rr;
		}else {
			ans+=abs(c[rr].x-c[nw].x);
			c[rr].ls=ll;
		}
	}
	printf("%d",ans);
	return 0;
}


你可能感兴趣的:(ac之路,数据结构,bzoj,平衡树)