@gym - 101137K@ Knights of the Old Republic
@description@
给定 N 个点 M 条边的一张图。
每个点有两个属性 Ai, Bi,表示你需要至少 Ai 个士兵来攻占该点,向 i 点投放一个士兵需要 Bi 的花费。
每条边都有一个属性 Ci,表示如果该边的两个端点的士兵数量之和 >= Ci,那么这条边就被打通了(即士兵可以自由通过该边)。
士兵不会死亡。求攻占所有点的最小代价和。
Input
第一行包含两个整数 n, m (1 ≤ n,m ≤ 300000)。
接下来 n 行,每行两个整数 ai 与 bi (0 ≤ ai,bi ≤ 1000000)。
接下来 m 行,每行三个整数 si, fi 与 ci 描述一条边 (si, fi) (1 ≤ si,fi ≤ n, 0 ≤ ci ≤ 1000000)。
Output
输出一个整数,表示最小花费。
Example
standard input
3 2
10 5
20 10
10 3
1 2 22
2 3 200
standard output
140
@solution@
最终局面下,对于一个连通块,它对答案的贡献应为 min{bi} * max{max{ai}, max{cj}}。
因为我至少要投放 max{max{ai}, max{cj}} 这么多士兵才能保证连通且攻占全部点,那不如直接投放这个连通块费用最小的点那里。
当连通块对应的点集不变时,max{cj} 应尽量小才能获得更优的答案。
在 max{cj} 最小的情况下连通这个连通块,这显然就是最小生成树了(参考 kruskal 的算法过程)。
继续分析。当一个连通块的 max{ai, cj} 大于它邻接的某条边 k 的 ck 时,我加入 k 这条边显然不会更劣(max{ai, cj} 不变,min{b} 反而可能变小)。
还可以得到一个更严格的限制:当一个连通块的 max{cj} 大于它邻接的某条边 k 的 ck 时,加入 k 这条边依然不会更劣。
正确性,简单来说,就是 A*B + C*D >= min(A, C)*max(B, D)。假如 B <= D,则 min(A, C)*max(B, D) = min(A, C)*D <= C*D <= A*B + C*D。
那么考虑 ci 最大的那一条边。假如它被选中,根据上面的理论,其他边也会被选中。如果选中其他边产生了环,证明它不是一棵最小生成树,显然不合法。否则,我们可以算出此时的答案。
假如它不被选中,相当于求去除了这条边的最优值。这又递归成一个子问题。
但是这个太慢了。不如反过来,考虑 ci 从小到大枚举。
假如此时加入这条边,产生了环,与最小生成树矛盾,此时直接忽视第 i 条边(类 kruskal 的过程)。
否则,计算这条边连接的两个点集的 min{b}, max{a}。此时因为 i 的 ci 是最大的,所以直接取 max{ci, max{a}} 即可。
为了与不加入第 i 条边的最优值对比,我们对于每个点集再维护个 f,表示加入前 i-1 条边该点集的最优答案是什么。
@accepted code@
#include<cstdio>
#include<algorithm>
using namespace std;
typedef long long ll;
const int MAXN = 300000;
struct edge{
int u, v; ll w;
edge(int _u=0, int _v=0, ll _w=0) : u(_u), v(_v), w(_w) {}
friend bool operator < (edge a, edge b) {
return a.w < b.w;
}
}e[MAXN + 5];
int fa[MAXN + 5];
ll f[MAXN + 5], ma[MAXN + 5], mb[MAXN + 5];
int find(int x) {
return fa[x] = (fa[x] == x ? x : find(fa[x]));
}
void unite(int x, int y, ll w) {
int fx = find(x), fy = find(y);
if( fx != fy ) {
mb[fy] = min(mb[fy], mb[fx]);
ma[fy] = max(ma[fy], ma[fx]);
f[fy] = min(f[fx] + f[fy], max(ma[fy], w)*mb[fy]);
fa[fx] = fy;
}
}
int main() {
int n, m; scanf("%d%d", &n, &m);
for(int i=1;i<=n;i++) scanf("%lld%lld", &ma[i], &mb[i]);
for(int i=1;i<=m;i++) scanf("%d%d%lld", &e[i].u, &e[i].v, &e[i].w);
for(int i=1;i<=n;i++) fa[i] = i, f[i] = ma[i]*mb[i];
sort(e + 1, e + m + 1);
for(int i=1;i<=m;i++) unite(e[i].u, e[i].v, e[i].w);
ll ans = 0;
for(int i=1;i<=n;i++)
if( find(i) == i ) ans += f[i];
printf("%lld\n", ans);
}
@details@
感觉还是比较巧妙的题,运用了比较多的性质,写出来代码也不是很长。
(我没学过图论.jpg)
@gym - 101137K@ Knights of the Old Republic的更多相关文章
- 2016 NEERC, Moscow Subregional Contest K. Knights of the Old Republic(Kruskal思想)
2016 NEERC, Moscow Subregional Contest K. Knights of the Old Republic 题意:有一张图,第i个点被占领需要ai个兵,而每个兵传送至该 ...
- 【Gym101137K】Knights of the Old Republic(生成树 DP)
题目链接 大意 给定\(N\)个点\(M\)条边的一张图,其中: 每个点有两个属性\(A_i,B_i\),表示你需要至少\(A_i\)个士兵来攻占该点,而空投一个士兵至该点需要Bi的花费. 每条边都有 ...
- 2016-2017 ACM-ICPC, NEERC, Moscow Subregional Contest
A. Altitude 从小到大加入每个数,用set查找前驱和后继即可. 时间复杂度$O(n\log n)$. #include <bits/stdc++.h> using namespa ...
- 3d引擎列表
免费引擎 Agar - 一个高级图形应用程序框架,用于2D和3D游戏. Allegro library - 基于 C/C++ 的游戏引擎,支持图形,声音,输入,游戏时钟,浮点,压缩文件以及GUI. A ...
- Codeforces Gym 100650D Queens, Knights and Pawns 暴力
Problem D: Queens, Knights and PawnsTime Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://acm.hust.edu ...
- Kernel Knights (Gym - 101480K)
题目链接 #include <bits/stdc++.h> using namespace std; typedef long long ll; int a[200005]; //存放原始 ...
- Gym - 101480K_K - Kernel Knights (DFS)
题意:有两队骑士各n人,每位骑士会挑战对方队伍的某一个位骑士. (可能相同) 要求找以一个区间s: 集合S中的骑士不会互相挑战. 每个集合外的骑士必定会被集合S内的某个骑士挑战. 题解:讲真被题目绕懵 ...
- Codeforces Gym100812 L. Knights without Fear and Reproach-扩展欧几里得(exgcd)
补一篇以前的扩展欧几里得的题,发现以前写错了竟然也过了,可能数据水??? 这个题还是很有意思的,和队友吵了两天,一边吵一边发现问题??? L. Knights without Fear and Rep ...
- ACM: Gym 101047M Removing coins in Kem Kadrãn - 暴力
Gym 101047M Removing coins in Kem Kadrãn Time Limit:2000MS Memory Limit:65536KB 64bit IO Fo ...
随机推荐
- 使用原生ajax及其简单封装
原生ajax配置详解 // 原生ajax // 1. 创建ajax对象 if(window.XMLHttpRequest){ // // IE7+, Firefox, Chrome, Opera, S ...
- Mocha测试
mocha中文名叫做摩卡,是javascript测试的一种常见手段. 其他的类似的测试还有jasmine.karma.tape等. 1. 测试脚本怎么写 // add.js function add( ...
- 怎么比较两个list中相同的值个数!
怎么比较两个list中相同的值个数!int count=0;for(int i=0;i<list1.size();i++){ for(int j=0;j<list2.size();j++) ...
- ORACLE的Copy命令和create table,insert into的比较
在数据表间复制数据是Oracle DBA经常面对的任务之一,Oracle为这一任务提供了多种解决方案,SQL*Plus Copy 命令便是其中之一.SQL*Plus Copy 命令通过SQL*Net在 ...
- 微信网页授权demo1
要授权首先要网页域名授权 然后就index.php代码如下 <?php require_once("./function.php"); $url = 'http://'.$_ ...
- laravel-admin
laravel-admin 文档地址: http://laravel-admin.org/docs/#/zh/
- python3.7的celery报错TypeError: wrap_socket() got an unexpected keyword argument '_context'
原启动方法为: 起执行任务的服务 elery worker -A celery_task -l info -P eventlet 起提交任务的服务 celery beat -A celery_task ...
- vue中 表头 th 合并单元格,且表格列数不定的动态渲染方法
吐槽 今天,在vue中遇到 复杂表格的渲染 ,需要合并表头th的单元格,且合并单元格的那列的表头数据是动态数据,也就是不知道会有多少个表头列,而这几个表头列还分了好几个子表头. 这个需求在js里用Ju ...
- JS DOM节点的增删改查
合并拆分 行内样式 script写在html里面
- 利用Eclipse进行远程Debug
这项功能真的十分赞,当我不想写junit test,又想调试在实际环境中才能起作用的Java程序,远程debug真的是太好用了. 参数:java -jar -Xdebug -Xrunjdwp:tran ...