【ARC098F】Donation
【ARC098F】Donation
题面
题意:
给定一张\(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的更多相关文章
- 【OS】NMON的简介和使用
[OS]NMON的简介和使用 目前NMON已开源,以sourceforge为根据地,网址是http://nmon.sourceforge.net. 1. 目的 本文介绍操作系统监控工具Nmon的概念. ...
- Python高手之路【六】python基础之字符串格式化
Python的字符串格式化有两种方式: 百分号方式.format方式 百分号的方式相对来说比较老,而format方式则是比较先进的方式,企图替换古老的方式,目前两者并存.[PEP-3101] This ...
- 【原】谈谈对Objective-C中代理模式的误解
[原]谈谈对Objective-C中代理模式的误解 本文转载请注明出处 —— polobymulberry-博客园 1. 前言 这篇文章主要是对代理模式和委托模式进行了对比,个人认为Objective ...
- 【原】FMDB源码阅读(三)
[原]FMDB源码阅读(三) 本文转载请注明出处 —— polobymulberry-博客园 1. 前言 FMDB比较优秀的地方就在于对多线程的处理.所以这一篇主要是研究FMDB的多线程处理的实现.而 ...
- 【原】Android热更新开源项目Tinker源码解析系列之一:Dex热更新
[原]Android热更新开源项目Tinker源码解析系列之一:Dex热更新 Tinker是微信的第一个开源项目,主要用于安卓应用bug的热修复和功能的迭代. Tinker github地址:http ...
- 【调侃】IOC前世今生
前些天,参与了公司内部小组的一次技术交流,主要是针对<IOC与AOP>,本着学而时习之的态度及积极分享的精神,我就结合一个小故事来初浅地剖析一下我眼中的“IOC前世今生”,以方便初学者能更 ...
- Python高手之路【三】python基础之函数
基本数据类型补充: set 是一个无序且不重复的元素集合 class set(object): """ set() -> new empty set object ...
- Python高手之路【一】初识python
Python简介 1:Python的创始人 Python (英国发音:/ˈpaɪθən/ 美国发音:/ˈpaɪθɑːn/), 是一种解释型.面向对象.动态数据类型的高级程序设计语言,由荷兰人Guido ...
- 【开源】简单4步搞定QQ登录,无需什么代码功底【无语言界限】
说17号发超简单的教程就17号,qq核审通过后就封装了这个,现在放出来~~ 这个是我封装的一个开源项目:https://github.com/dunitian/LoTQQLogin ————————— ...
随机推荐
- jmeter(二十七)分布式压测注意事项
之前的博客:jemter(二十三):分布式测试简略的介绍了利用jmeter做分布式测试的方法,当时只是介绍了背景和原因,以及基本的配置操作,有同学说写得不够详细. 正好今年双十一,我司的全链路压测,也 ...
- Eclipse中js文件修改后浏览器不能及时更新的解决办法
项目中js文件修改后浏览器不能及时更新的解决办法 转载:http://www.codeweblog.com/%E9%A1%B9%E7%9B%AE%E4%B8%ADjs%E6%96%87%E4%BB%B ...
- 创建你自己定制的vuejs plugin扩展app的功能
什么是vuejs plugin插件 vuejs plugin插件是一个向你的app注入新的全局功能的强大但又简约的方式.从概念上来说,vue plugin非常简单,它就是一个包含了install方法的 ...
- Java自学-类和对象 类方法
Java的类方法和对象方法 类方法: 又叫做静态方法 对象方法: 又叫实例方法,非静态方法 访问一个对象方法,必须建立在有一个对象的前提的基础上 访问类方法,不需要对象的存在,直接就访问 步骤 1 : ...
- vue组件4 利用slot将内容传递给组件
除了将数据作为prop传入到组件中,vue也允许传入HTML 父组件中的子组件:<custom-button>点我<custom-button/> custom-button子 ...
- Java 之 Random 类
一.Random 类 random 类的实例用于生成伪随机数. Demo: Random r = new Random(); int i = r.nextInt(); 二.Random 使用步骤 1 ...
- url请求时,参数中的+在服务器接收时为空格,导致AES加密报出javax.crypto.IllegalBlockSizeException: Input length must be multiple of 16 when decrypting with padded cipher
报错的意思的是使用该种解密方式出入长度应为16bit的倍数,但实际的错误却不是这个,错误原因根本上是因为在http请求是特殊字符编码错误,具体就是base64生成的+号,服务器接收时成了空格,然后导致 ...
- 百度地图公交线路查询,并绘制到地图上并获取所有路径经纬度点(可供echarts 路径图使用)
github地址 https://github.com/a1115040996/MyHTML/tree/gh-pages/BDMap 源代码 <!DOCTYPE html PUBLIC &quo ...
- LINUX基础学习之基础命令(4)和 文件的inode信息 --2019-11-28
一.inode是什么? 理解inode,要从文件储存说起. 文件储存在硬盘上,硬盘的最小存储单位叫做”扇区”(Sector).每个扇区储存512字节(相当于0.5KB). 操作系统读取硬盘的时候,不会 ...
- 网络空间安全基础篇(1)----nmap的使用
nmap工具通俗一点说,就是利用nmap工具来对靶机的所有端口及其开启的服务进行扫描,以便于黑客入侵靶机时,知道用什么漏洞去攻击. nmap常用的命令集(标为红色的一般为常用参数): sV 扫描版本O ...