代码风格与树形DP

Streaming很惨,不过因为比赛之间没有提交过就没掉(或掉了)rating.第二题是一个树形DP,但是我都在想第一题了,简直作死.

看着神犇的代码我也是醉了...各种宏,真是好好写会死系列. 看到他们Tree DP都用的DFS,突然感觉我这个蒟蒻的生活中充满了无力...

我一般都喜欢用BFS进行Tree DP.这样坏处很多,难调试,容易爆空间等.好处也有,写起来快,代码短,跑得飞快,判重简单.不过这样做是有条件的,而今天的Streaming这题就是一道可以的题.

然后讲讲今天这道LCAStat吧.

作为一个不会写Tarjan LCA的蒟蒻,这道题出的比较有素质.至少你不需会LCA(树上倍增差不多会吧).有些神犇这么暴力,对此我只能表示Orz.

让我们思考一下.对于每一个点,他的(姑且称为LCA Score)应该怎么算?

转换个思路,如何让两个点的LCA为一个给定的点i?

显然,这两个点必须是i或在i的不同子树中.那么我们就渐渐有了思路.

注意算sum(x){sum(y){x*y}}有个很简单的办法即sum(x){x}*sum(y){y}

那么求出每一棵子树的totalWeight,当在一个结点新访问到一个子结点时,这个点的对分值贡献就是totw[i]*(noww[d]+w[d])*2

totw[i]就是这个访问到的子结点的totalWeight,noww[d]就是已经加入这个父结点(不包括新访问的这个)的子树的总重,w[d]就是这个父结点的weight

为什么要乘以2?因为这道题目考虑顺序,即a=3,b=5和a=5,b=3要算两次.

每次有一个结点的所有子结点都被访问了就松弛这个结点.

自己看代码吧,去了所有mod.

#include <cstdio>
#include <cstring>
long long fat[200000],w[200000],f[200000],sub[200000],totw[200000],n,p,i,sum;
long long q[200000],qh,qt;
int main(int argc,char const *argv[]){
	scanf("%lld %lld",&n,w+1);
	for(i=2;i<=n;++i){
		scanf("%lld %lld",fat+i,w+i);
		++sub[fat[i]];
	}
	for(i=1;i<=n;++i){
		f[i]=0;
		if(!sub[i]){
			q[qt++]=i;
			totw[i]=0;
			f[i]=0;
		}
	}
	while(qh!=qt){
		i=q[qh++];
		f[i]+=(w[i]*w[i])*(w[i]+totw[i]*2);
		totw[i]+=w[i];
		sum=sum+f[i];
		f[fat[i]]+=(totw[i]*totw[fat[i]])*w[fat[i]]*2;
		totw[fat[i]]+=totw[i];
		--sub[fat[i]];
		if(!sub[fat[i]]) q[qt++]=fat[i];
	}
	printf("%lld\n", sum);
	return 0;
}
//非AC代码,只是为了清楚的一份演示代码

  

posted @ 2014-10-04 23:30  zball  阅读(226)  评论(0编辑  收藏  举报