Luogu P3408 恋爱

题目链接:Click here

Solution:

设f[x]表示要使x向它的父亲写信需要花的最少的钱,per[x]为要使x向它的父亲写信最少要多少人

\(f[x]=\sum_{i=1}^{per[x]}f[son[x]]\),此时的f数组是从小到大排过序的

那我们只需要把每个点的儿子放到multiset去维护就好了,最后输出f[0]

总时间复杂度\(O(n\,log\,n)\)

Code:

#include<bits/stdc++.h>
#define int long long
using namespace std;
const int N=5e5+1;
const int inf=2147483647;
int n,m,t,cnt,head[N];
int ans,a[N],per[N],cost[N],num[N];
struct Edge{int nxt,to;}edge[N];
struct cmp{bool operator()(const int& a,const int& b)const{return cost[a]<cost[b];}};
multiset<int,cmp> q[N];
multiset<int,cmp>::iterator it;
void ins(int x,int y){
	edge[++cnt].nxt=head[x];
	edge[cnt].to=y;head[x]=cnt;
}
void dfs(int x){
	int flag=0;
	for(int i=head[x];i;i=edge[i].nxt){
		int y=edge[i].to;
		dfs(y);q[x].insert(y);
		flag=1;
	}
	if(!flag){cost[x]=a[x];return ;}
	if(x){
		int u=per[x];it=q[x].begin();
		for(int i=1;i<=u;i++){
			if(it==q[x].end()){
				cost[x]=inf;
				return ;
			}cost[x]+=cost[*it],++it;
		}
	}else{int w=(m*num[x]+t-1)/t;
		it=q[x].begin();
		for(int i=1;i<=w;i++)
			ans+=cost[*it],++it;
	}
}
int read(){
	int x=0,f=1;char ch=getchar();
	while(!isdigit(ch)){if(ch=='-')f=-1;ch=getchar();}
	while(isdigit(ch)){x=x*10+ch-48;ch=getchar();}
	return x*f;
}
signed main(){
	n=read(),t=read(),m=read();
	for(int i=1;i<=n;i++){
		int x=read(),y=read();
		ins(x,i);a[i]=y;num[x]++;
	}
	for(int i=1;i<=n;i++) per[i]=(num[i]*a[i]+t-1)/t;
	dfs(0);printf("%lld\n",ans);
	return 0;
}
posted @ 2019-07-05 21:28  DQY_dqy  阅读(148)  评论(1编辑  收藏  举报