[ARC098D] Donation

一、题目

点此看题

二、解法

考虑把原过程逆序,这样捐赠 \(b_i\) 就变成获得(贪污) \(b_i\) 元,经过一个点的条件是现有的钱数 \(\geq \max(a_i-b_i,0)=c_i\)(其实 \(a_i-b_i\) 是一个很重要的量,虽可以通过其他方式得出但不好继续走下去)

那么如果可以任意走,按照 \(c_i\) 从小到大的顺序是最优的。但是现在有无向图的限制,有一个关键的 \(\tt observation\):给边 \((u,v)\) 赋权值 \(\max(c_u,c_v)\),那么最优方案只会走最小生成树上的路径

这样看性质还不是特别明显,我们考虑 \(c_i\) 从小到大建立 \(\tt kruskal\) 重构树,这样某个点到根的路径上 \(c\) 一定是依次增大的。并且最优策略一定是从一个叶子开始,一路走到根,中途顺道经过其他子树。

那么可以考虑 \(dp\) 确定起点了,设 \(f_u\) 表示起点在子树 \(u\) 以内的最小初始代价。如果 \(u\) 是叶子那么 \(f_u=c_u\),最后的答案是 \(f_{rt}+sumb_{rt}\),转移:

\[f_u=\min_v\{\max(f_v,c_u-sumb_v)\} \]

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

#include <cstdio>
#include <vector>
#include <iostream>
#include <algorithm>
using namespace std;
const int M = 100005;
#define int long long
#define pb push_back
int read()
{
	int x=0,f=1;char c;
	while((c=getchar())<'0' || c>'9') {if(c=='-') f=-1;}
	while(c>='0' && c<='9') {x=(x<<3)+(x<<1)+(c^48);c=getchar();}
	return x*f;
}
int n,m,a[M],b[M],c[M],f[M],p[M],in[M],fa[M];
vector<int> g[M],G[M];
int find(int x)
{
	if(x==fa[x]) return x;
	return fa[x]=find(fa[x]);
}
void dfs(int u)
{
	f[u]=c[u];
	for(int v:G[u])
	{
		dfs(v);b[u]+=b[v];
		f[u]=min(f[u],max(f[v],c[u]-b[v]));
	}
}
signed main()
{
	n=read();m=read();
	for(int i=1;i<=n;i++)
	{
		a[i]=read();b[i]=read();
		c[i]=max(a[i]-b[i],0ll);
		p[i]=fa[i]=i;
	}
	for(int i=1;i<=m;i++)
	{
		int u=read(),v=read();
		g[u].pb(v);g[v].pb(u);
	}
	sort(p+1,p+1+n,[&](int i,int j)
		{return c[i]<c[j];});
	for(int i=1;i<=n;i++)
	{
		int u=p[i];in[u]=1;
		for(int v:g[u])
			if(in[v] && find(u)^find(v))
			{
				G[u].pb(find(v));
				fa[find(v)]=find(u);
			}
	}
	int rt=p[n];dfs(rt);
	printf("%lld\n",f[rt]+b[rt]);
}
posted @ 2022-04-08 20:05  C202044zxy  阅读(43)  评论(0编辑  收藏  举报