【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. c#读取数据库bool值

    数据库里bit这个布尔类型的字段,非常实用,但是在c#里读取时,许多人喜欢犯一些错误,导致运行报错. 实际中,有效的正确读取方法只有以下两种: int xxx= Convet.ToInt16(read ...

  2. Ubuntu系统下基于docker部署Jenkins环境

    本文是在ubuntu环境下安装jenkins,jenkins运行在docker容器中, 至于docker如何安装,请参考https://www.cnblogs.com/xingyunqiu/p/115 ...

  3. Vue安装及项目介绍

    目录 创建Vue项目 环境安装 创建项目 pycharm打开Vue项目 项目目录介绍 入口文件(main.js) 路由配置(router.js ) 组件 前台路由的基本工作流程 目录结构 根组件(Ap ...

  4. Spring Boot 框架下使用MyBatis访问数据库之基于XML配置的方式

    MyBatis 是一款优秀的持久层框架,它支持定制化 SQL.存储过程以及高级映射.MyBatis 避免了几乎所有的 JDBC 代码和手动设置参数以及获取结果集.MyBatis 可以使用简单的 XML ...

  5. 分布式事务:Saga模式

    1 Saga相关概念 1987年普林斯顿大学的Hector Garcia-Molina和Kenneth Salem发表了一篇Paper Sagas,讲述的是如何处理long lived transac ...

  6. 查看Linux内核版本

    您可能因多种原因需要确切知道GNU / Linux操作系统上运行的内核版本. 也许您正在调试与硬件相关的问题,或者了解影响旧内核版本的新安全漏洞,并且您想知道您的内核是否易受攻击. 无论是什么原因,从 ...

  7. English--不定式

    English|不定式 不定式短语在英语的写作中,比较常见,所以需要多留意.英语的学习需要多加阅读,在阅读中,看句子,学语法. 前言 目前所有的文章思想格式都是:知识+情感. 知识:对于所有的知识点的 ...

  8. CDN: trunk Repo update failed - CocoaPods

    解决方案: 1.podfile文件中添加source源:  source 'https://github.com/CocoaPods/Specs.git' 2.执行 pod repo remove t ...

  9. 你的MES今天升级了吗?

    你以为把MES装上了就完事了吗?NO NO NO!乔布斯先生曾讲过“你如果出色地完成了某件事,那你应该再做一些其他的精彩事儿.不要在前一件事上徘徊太久,想想接下来该做什么.” 目前大部分企业都已经完成 ...

  10. Python如何去实际提高工作的效率?也许这个会有用!

    4月初,班主任的某次周会议上,华华关切的问了一下:最近班主任们有什么难题吗?就是花费了你们大部分时间的工作!我们Python天团可以帮你们解决问题. 班主任大主管星星说:有.目前有一个大难题.我们每天 ...