首先我们强制要求几条待定价的边在MST中,建出MST

我们发现这个MST中原来的边是一定要被选上的,所以可以把点缩起来,搞成一棵只有$K$个点的树

然后$2^K$枚举每条边在不在最终的MST中,让在最终MST中的待定价的边尽量大,只需要在Kruskal的时候暴力更新每条边的定价即可

时间复杂度$O(m * logm + 2^K * K^2)$

 /**************************************************************
Problem: 3206
User: rausen
Language: C++
Result: Accepted
Time:8040 ms
Memory:8232 kb
****************************************************************/ #include <cstdio>
#include <algorithm> using namespace std;
typedef long long ll;
const int N = 1e5 + ;
const int M = 3e5 + ;
const int K = ;
const int inf = 1e9; inline int read(); struct Edge {
int x, y, v; inline void get(int f) {
x = read(), y = read();
if (f) v = read();
} inline bool operator < (const Edge &E) const {
return v < E.v;
}
} E[M], Ek[K], s[K]; struct edge {
int next, to;
edge() {}
edge(int _n, int _t) : next(_n), to(_t) {}
} e[K << ]; int first[N], tot; struct tree_node {
int fa, dep, mn;
ll v, sum;
} tr[N]; int n, m, k, S, top;
int fa[][N];
int root[K], cnt_root;
int u[K];
ll ans; inline void Add_Edges(int x, int y) {
e[++tot] = edge(first[x], y), first[x] = tot;
e[++tot] = edge(first[y], x), first[y] = tot;
} int find(int x, int f) {
return x == fa[f][x] ? x : fa[f][x] = find(fa[f][x], f);
} #define y e[x].to
void dp(int p) {
int x;
tr[p].sum = tr[p].v;
for (x = first[p]; x; x = e[x].next)
if (y != tr[p].fa) {
tr[y].dep = tr[p].dep + , tr[y].fa = p;
dp(y);
tr[p].sum += tr[y].sum;
}
}
#undef y ll work() {
static int i, x, y, p;
static ll res;
for (tot = , i = ; i <= k + ; ++i) {
p = root[i];
fa[][p] = p;
first[p] = tr[p].fa = , tr[p].mn = inf;
}
for (i = ; i <= k; ++i)
if (u[i]) {
x = find(Ek[i].x, ), y = find(Ek[i].y, );
if (x == y) return ;
fa[][x] = y;
Add_Edges(Ek[i].x, Ek[i].y);
}
for (i = ; i <= k; ++i) {
x = find(s[i].x, ), y = find(s[i].y, );
if (x != y) fa[][x] = y, Add_Edges(s[i].x, s[i].y);
}
dp(S);
for (i = ; i <= k; ++i) {
x = s[i].x, y = s[i].y;
if (tr[x].dep < tr[y].dep) swap(x, y);
while (tr[x].dep != tr[y].dep)
tr[x].mn = min(tr[x].mn, s[i].v), x = tr[x].fa;
while (x != y) {
tr[x].mn = min(tr[x].mn, s[i].v);
tr[y].mn = min(tr[y].mn, s[i].v);
x = tr[x].fa, y = tr[y].fa;
}
}
#define x Ek[i].x
#define y Ek[i].y
for (res = , i = ; i <= k; ++i)
if (u[i])
res += tr[x].dep > tr[y].dep ? tr[x].mn * tr[x].sum : tr[y].mn * tr[y].sum;
#undef x
#undef y
return res;
} void dfs(int p) {
if (p == k + ) {
ans = max(ans, work());
return;
}
u[p] = , dfs(p + );
u[p] = , dfs(p + );
} int main() {
int i, x, y;
n = read(), m = read(), k = read();
for (i = ; i <= m; ++i) E[i].get();
for (i = ; i <= k; ++i) Ek[i].get();
for (i = ; i <= n; ++i) tr[i].v = read();
sort(E + , E + m + );
for (i = ; i <= n; ++i) fa[][i] = fa[][i] = i; for (i = ; i <= k; ++i)
fa[][find(Ek[i].x, )] = find(Ek[i].y, );
#define x E[i].x
#define y E[i].y
for (i = ; i <= m; ++i)
if (find(x, ) != find(y, ))
fa[][find(x, )] = fa[][find(y, )], fa[][find(x, )] = fa[][find(y, )];
#undef x
#undef y
S = find(, );
for (i = ; i <= n; ++i)
if (find(i, ) != i) tr[find(i, )].v += tr[i].v;
else root[++cnt_root] = i;
#define x Ek[i].x
#define y Ek[i].y
for (i = ; i <= k; ++i)
x = find(x, ), y = find(y, );
#undef x
#undef y
#define x E[i].x
#define y E[i].y
for (i = ; i <= m; ++i)
x = find(x, ), y = find(y, );
for (i = ; i <= m; ++i)
if (find(x, ) != find(y, ))
s[++top] = E[i], fa[][find(x, )] = find(y, );
#undef x
#undef y
dfs();
printf("%lld\n", ans);
return ;
} inline int read() {
static int x;
static char ch;
x = , ch = getchar();
while (ch < '' || '' < ch)
ch = getchar();
while ('' <= ch && ch <= '') {
x = x * + ch - '';
ch = getchar();
}
return x;
}

BZOJ3206 [Apio2013]道路费用的更多相关文章

  1. [BZOJ3206][APIO2013]道路费用(最小生成树)

    3206: [Apio2013]道路费用 Time Limit: 20 Sec  Memory Limit: 128 MBSubmit: 568  Solved: 266[Submit][Status ...

  2. [Bzoj3206][Apio2013]道路费用(kruscal)(缩点)

    3206: [Apio2013]道路费用 Time Limit: 20 Sec  Memory Limit: 128 MBSubmit: 536  Solved: 252[Submit][Status ...

  3. 洛谷P3639 [APIO2013] 道路费用 [生成树的特殊算法]

    题目传送门 道路费用 格式难调,题面就不放了. 分析: 这是一道要细(yan)心(jing)的生成树的好(gui)题. 首先我们看到$k$的范围非常小,那么我们就可以直接$2^k$枚举每一条加边是否选 ...

  4. [APIO2013]道路费用

    题目描述 幸福国度可以用 N 个城镇(用 1 到 N 编号)构成的集合来描述,这些城镇 最开始由 M 条双向道路(用 1 到 M 编号)连接.城镇 1 是中央城镇.保证一个 人从城镇 1 出发,经过这 ...

  5. 题解 [APIO2013]道路费用

    link Description 幸福国度可以用 N 个城镇(用 1 到 N 编号)构成的集合来描述,这些城镇 最开始由 M 条双向道路(用 1 到 M 编号)连接.城镇 1 是中央城镇.保证一个 人 ...

  6. bzoj 3206: [Apio2013]道路费用【最小生成树+并查集】

    参考:http://hzwer.com/6888.html 把k条道路权值设为0,和其他边一起跑MST,然后把此时选中的其他边设为必选,在新图中加上必选变缩成k个点,把所有边重标号,枚举k跳边的选取情 ...

  7. 题解 洛谷 P3639 【[APIO2013]道路费用 】

    不难想到可以\(2^k\)去枚举\(k\)条新边的选择方案,然后加入原图中的边来使图连通,用当前方案的收益去更新答案,但是这样复杂度过不去. 可以先把\(k\)条新边都连上,然后再加入边权从小到大排序 ...

  8. PKUSC2018训练日程(4.18~5.30)

    (总计:共66题) 4.18~4.25:19题 4.26~5.2:17题 5.3~5.9: 6题 5.10~5.16: 6题 5.17~5.23: 9题 5.24~5.30: 9题 4.18 [BZO ...

  9. [APIO2013]

    A.机器人 题目大意:给定一个n*m的地图,有一些障碍物和k个机器人,你每次可以选择一个机器人往任意一个方向推,遇到转向器会转向,两个编号相邻的机器人可以合并,求最少推多少次可以全部合并. $n,m\ ...

随机推荐

  1. wordpress中文标签无法访问的解决方法

    wordpress中文标签无法访问的解决方法  爱好  2年前 (2014-05-29)  7,601  8 当博客从华夏名网转移到阿里云之后,发现了不少问题,其中一个就是wordpress中文标签无 ...

  2. python_way ,day5 模块,模块3 ,双层装饰器,字符串格式化,生成器,递归,模块倒入,第三方模块倒入,序列化反序列化,日志处理

    python_way.day5 1.模块3 time,datetime, json,pickle 2.双层装饰器 3.字符串格式化 4.生成器 5.递归 6.模块倒入 7.第三方模块倒入 8.序列化反 ...

  3. bzoj 2242: [SDOI2011]计算器 BSGS+快速幂+扩展欧几里德

    2242: [SDOI2011]计算器 Time Limit: 10 Sec  Memory Limit: 512 MB[Submit][Status][Discuss] Description 你被 ...

  4. Codeforces Round #375 (Div. 2) D. Lakes in Berland dfs

    D. Lakes in Berland time limit per test 2 seconds memory limit per test 256 megabytes input standard ...

  5. scala中如何编写自定义的流程控制结构

    scala是一种函数式编程风格的语言,除了常见的if......else  ,for ,while等传统的流程控制结构,也可以自定义流程控制的控制结构. 再了解scala如何实现编写新的流程结构,我们 ...

  6. Oracle VM VirtualBox 虚拟机与主机共享文件

    安装增强功能(参考文档) VirtualBox自带了一个增强工具Sun VirtualBox Guest Additions,这是实现虚拟机与真实主机共享的关键.启动虚拟机后,点击控制菜单“设备”→“ ...

  7. iOS - OC 内存管理

    1.OC 基本内存管理模型 1.1 自动垃圾收集 在 OC 2.0 中,有一种称为垃圾收集的内存管理形式.通过垃圾收集,系统能够自动监测对象是否拥有其他的对象,当程序执行需要空间的时候,不再被引用的对 ...

  8. EF中使用SQL语句或存储过程(小笔记)

    1.无参数查询 var model = db.Database.SqlQuery<UserInfo>("select* from UserInfoes ").ToLis ...

  9. 如何制作exe程序可执行文件

    很多软件的运行都需要搭建环境,只有exe文件可以在不安装软件和数据库的环境下运行,那么怎么制作exe程序可执行文件呢,下面天使教你如何制作.   工具/原料   Microsoft Visual St ...

  10. Android ViewFlipper控件实例

    使用ViewFlipper实现两张图片切换效果,废话不多说,直接上代码. java源码: package com.example.viewflipper; import android.os.Bund ...