• [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]);
    }
    
  • 相关阅读:
    《活在恩典中》第一章 人类的两难困境
    《真正的修行》把你内心的一切都呈现出来
    Mysql:Plugin:clone=mysql_clone:as of 8.0.17
    Mysql:--init-file && --init-connect
    Mysql:8.0.19:Upgrading Mysql:升级
    Android开发自定义View
    Android控制UI界面
    Android的视图(View)组件
    对Android应用签名
    Android Application的基本组件介绍
  • 原文地址:https://www.cnblogs.com/C202044zxy/p/16119545.html
Copyright © 2020-2023  润新知