[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]);
}