【ARC098F】Donation

题面

atcoder

题意:

给定一张\(n\)个点,\(m\)条边的无向图。这张图的每个点有两个权值 \(a_i,b_i\)。

你将会从这张图中选出一个点作为起点,随后开始遍历这张图。

你能到达一个节点 \(i\)当且仅当你的手上有至少\(a_i\)元钱。当你到达一个节点\(i\) 后,你可以选择对这个点捐赠\(b_i\)元。

你需要对每个点捐赠一次。问你身上至少要带多少元钱?

其中\(1\leq n\leq 10^5\),\(n-1\leq m\leq 2\times 10^5\)。

题解

首先你需要知道的两个性质:

  • 对于一个点,你只有最后访问它时才会对它进行捐赠。

    证明:这一点正确性比较显然。
  • 定义权值\(w=max(a_i-b_i,0)\),那么在满足上面条件的前提下你要尽量让\(w\)大的点先捐赠。

    证明:首先你要明白这个\(w\)是怎么来的,

    当你最后经过一个点并捐赠它时,

    你所剩下的钱\(w+b_i\geq a_i\),

    那么必须满足\(w\geq max(a_i-b_i,0)\)。

    所以你在最后访问这个点你所剩的钱最多,所以尽量先捐赠这个点。

考虑有了上述两点之后我们怎么解决问题,

倘若一个点在一个连通块内而且我们不需要再访问这个点,

那么这个点对答案影响不大,但是若一个点把联通块割成许多个小联通块,那就另当别论了。

基于这一点首先我们按权值\(w\)从小到大构建一颗生成树,

一边构建一边求答案(没必要真的建树,建树的过程可用并查集维护)。

设\(dp_i\)表示以\(i\)为根的子树所需的最小钱数,

\(sum_i\)表示子树\(i\)中所有的\(b_i\)之和,

则我们当一个点是叶子节点是答案就是\(dp_i=w_i+b_i\),

对于非叶子节点,我们枚举访问完子树\(j\)就不再回到\(i\)了,

根据我们上面的描述,转移就是\(dp_i=\min_{j\in son_i} sum_i-sum_j+max(w_i,dp_j)\)。

代码

#include <iostream>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <cmath>
#include <algorithm>
#include <vector>
using namespace std;
inline int gi() {
register int data = 0, w = 1;
register char ch = 0;
while (!isdigit(ch) && ch != '-') ch = getchar();
if (ch == '-') w = -1, ch = getchar();
while (isdigit(ch)) data = 10 * data + ch - '0', ch = getchar();
return w * data;
}
const int MAX_N = 1e5 + 5;
int N, M, a[MAX_N], b[MAX_N], p[MAX_N];
int pa[MAX_N];
int getf(int x) { while (x != pa[x]) x = pa[x] = pa[pa[x]]; return x; }
vector<int> G[MAX_N];
long long f[MAX_N], sum[MAX_N];
bool vis[MAX_N];
int main () {
#ifndef ONLINE_JUDGE
freopen("cpp.in", "r", stdin);
#endif
N = gi(), M = gi();
for (int i = 1; i <= N; i++) {
a[i] = gi(), b[i] = gi();
a[i] = max(a[i] - b[i], 0);
p[i] = i, pa[i] = i;
}
sort(&p[1], &p[N + 1], [](const int &l, const int &r) { return a[l] < a[r]; } );
for (int i = 1; i <= M; i++) {
int u = gi(), v = gi();
G[u].push_back(v), G[v].push_back(u);
}
for (int i = 1; i <= N; i++) {
vector<int> son;
int x = p[i];
vis[x] = 1, sum[x] = b[x];
for (auto v : G[x]) {
if (!vis[v] || getf(x) == getf(v)) continue;
son.push_back(getf(v));
sum[x] += sum[getf(v)];
pa[getf(v)] = x;
}
f[x] = sum[x] + a[x];
for (auto v : son) f[x] = min(f[x], sum[x] - sum[v] + max(1ll * a[x], f[v]));
}
printf("%lld\n", f[p[N]]);
return 0;
}

【ARC098F】Donation的更多相关文章

  1. 【OS】NMON的简介和使用

    [OS]NMON的简介和使用 目前NMON已开源,以sourceforge为根据地,网址是http://nmon.sourceforge.net. 1. 目的 本文介绍操作系统监控工具Nmon的概念. ...

  2. Python高手之路【六】python基础之字符串格式化

    Python的字符串格式化有两种方式: 百分号方式.format方式 百分号的方式相对来说比较老,而format方式则是比较先进的方式,企图替换古老的方式,目前两者并存.[PEP-3101] This ...

  3. 【原】谈谈对Objective-C中代理模式的误解

    [原]谈谈对Objective-C中代理模式的误解 本文转载请注明出处 —— polobymulberry-博客园 1. 前言 这篇文章主要是对代理模式和委托模式进行了对比,个人认为Objective ...

  4. 【原】FMDB源码阅读(三)

    [原]FMDB源码阅读(三) 本文转载请注明出处 —— polobymulberry-博客园 1. 前言 FMDB比较优秀的地方就在于对多线程的处理.所以这一篇主要是研究FMDB的多线程处理的实现.而 ...

  5. 【原】Android热更新开源项目Tinker源码解析系列之一:Dex热更新

    [原]Android热更新开源项目Tinker源码解析系列之一:Dex热更新 Tinker是微信的第一个开源项目,主要用于安卓应用bug的热修复和功能的迭代. Tinker github地址:http ...

  6. 【调侃】IOC前世今生

    前些天,参与了公司内部小组的一次技术交流,主要是针对<IOC与AOP>,本着学而时习之的态度及积极分享的精神,我就结合一个小故事来初浅地剖析一下我眼中的“IOC前世今生”,以方便初学者能更 ...

  7. Python高手之路【三】python基础之函数

    基本数据类型补充: set 是一个无序且不重复的元素集合 class set(object): """ set() -> new empty set object ...

  8. Python高手之路【一】初识python

    Python简介 1:Python的创始人 Python (英国发音:/ˈpaɪθən/ 美国发音:/ˈpaɪθɑːn/), 是一种解释型.面向对象.动态数据类型的高级程序设计语言,由荷兰人Guido ...

  9. 【开源】简单4步搞定QQ登录,无需什么代码功底【无语言界限】

    说17号发超简单的教程就17号,qq核审通过后就封装了这个,现在放出来~~ 这个是我封装的一个开源项目:https://github.com/dunitian/LoTQQLogin ————————— ...

随机推荐

  1. Java 8 ArrayList 详解

    GitHub Page: http://blog.cloudli.top/posts/Java-ArrayList/ ArrayList 继承于 AbstractList ,实现了 List.Rand ...

  2. linux学习-防火墙指令

    Redhat7之前的版本(iptables) 开启关闭防火墙 放行端口 RedHat7防火墙相关的指令(firewall-cmd) 安装firewall 本文内容适用于 redhat 和 centos ...

  3. 将html中的内容生成PDF并且下载

    <head> @*需要引用的js库*@ <script src="https://cdnjs.cloudflare.com/ajax/libs/html2canvas/0. ...

  4. k-匿名算法

    30 November 2019 18:31     人类历史上,除了计算机外从没有一项技术可以在短短的几十年间,能够全方位的影响整个社会的各个领域.技术的发展,少不了许多代人为之的努力.无论是在计算 ...

  5. python3对字符串进行base64转码

    import base64# 使用base64的b64encode()进行转码,转码之后在用‘utf-8’解码# s 要转码的字符串res = base64.b64encode(s.encode(&q ...

  6. 对Apache2进行简单配置

    Apache2 1.安装Apache2 sudo apt-get update sudo apt-get install apache2 2.启动服务 sudo /etc/init.d/apache2 ...

  7. nodejs npm vue yarn 环境搭建

    1.nodejs中文网http://nodejs.cn/download/下载最新版本 2.安装注意,虽然.msi包没有右键用管理员权限运行,如果直接双击安装可能会存在安装失败的问题.使用管理员权限运 ...

  8. Float型 与 Double型数据的存储方式

    先来了解一下浮点数在计算机中是以什么形式存储的 首先要知道计算机能懂得只有0和1,每一个0和1都占一个位 bit (比特)(Binary Digits):存放一位二进制数,最小的存储单位. 而对于存放 ...

  9. Tachyon内存文件系统快速入门

    一.简介 Tachyon是介于磁盘存储和计算框架之间的一种中间件,用于实现分布式的内存文件读写等功能,实现分布式集群内部共享数据. 应用实例: 二.架构 1.心跳机制 在Tachyon中,心跳用于Ma ...

  10. OSI&TCP/IP模型

    OSI模型 OSI协议 基于OSI模型异构网络主机之间通信的协议 1. 应用层 不同的应用程序所在层,如邮件程序,web程序 2. 表示层 不同的文件类型如图片,视频等 3. 会话层 决定是否建立与其 ...