[BZOJ 1937][Shoi2004]Mst 最小生成树
$ \color{red} {solution:} $
对于每条树边\(i\),其边权只可能变小,对于非树边\(j\),其边权只可能变大,所以对于任意非树边覆盖的树边有 \(wi - di <= wj + dj\),
变形一下 \(wi - wj <= di +dj\), 而这一部分正是带权二分图匹配,可以使用\(KM\)算法
不会\(KM\)算法的同学这里有篇算法介绍;
#include <bits/stdc++.h>
using namespace std;
const int maxn = 10010, inf = 0x3f3f3f3f;
template <typename T> inline bool check_Max(T &x, const T &y) {
return x < y ? x = y, false : true;
}
template <typename T> inline bool check_Min(T &x, const T &y) {
return x > y ? x = y, false : true;
}
struct edge {int u, v, w; } E[maxn];
int v[maxn], to[maxn], head[maxn], pos[maxn], p;
inline void link(int a, int b, int c) {
v[++ p] = b; to[p] = head[a]; head[a] = p; pos[p] = c;
}
inline void init() {
p = 1; memset(head, 0, sizeof(head));
}
int dep[maxn], fa[maxn], rfe[maxn];
inline void dfs(int u, int f) {
dep[u] = dep[f] + 1; fa[u] = f;
for ( int i = head[u]; i; i = to[i]) if( v[i] ^ f)
rfe[v[i]] = i >> 1, dfs(v[i], u);
}
int a[1010][1010], pre[maxn], slack[maxn], vis[maxn], mat[maxn], tot;
int lv[maxn], rv[maxn];
inline void aug(int s) {
for ( int i = 0; i <= tot; ++ i) slack[i] = inf, vis[i] = pre[i] = 0;
int u = 0; mat[u] = s;
do {
int now = mat[u], d = inf, nxt; vis[u] = 1;
for ( int v = 1; v <= tot; ++ v) if( !vis[v]){
if( !check_Min(slack[v], lv[now] + rv[v] - a[now][v]))
pre[v] = u;
if( !check_Min(d, slack[v])) nxt = v;
}
for ( int i = 0; i <= tot; ++ i)
if( vis[i]) lv[mat[i]] -= d, rv[i] += d;
else slack[i] -= d;
u = nxt;
} while ( mat[u]);
while ( u) mat[u] = mat[pre[u]], u = pre[u];
}
int n, m;
int main() {
#ifndef ONLINE_JUDGE
freopen("1.in","r",stdin);
#endif
init();
scanf("%d%d", &n, &m); tot = max(n-1, m-n+1);
for ( int i = 1; i <= m; ++ i) {
scanf("%d%d%d", &E[i].u, &E[i].v, &E[i].w);
if( E[i].u > E[i].v) swap(E[i].u, E[i].v);
}
for ( int i = 1, u, v; i < n; ++ i) {
scanf("%d%d", &u, &v); if( u > v) swap(u, v);
for ( int k = 1; k <= m; ++ k) if( E[k].u == u && E[k].v == v) {
vis[k] = 1, link(u, v, E[k].w), link(v, u, E[k].w); break;
}
}
dfs(1, 0);
int num = 0;
for ( int i = 1; i <= m; ++ i) if( !vis[i]) {
int u = E[i].u, v = E[i].v, c; ++ num;
while ( u ^ v) {
if( dep[u] < dep[v]) swap(u, v);
c = rfe[u];
a[c][num] = pos[c << 1] - E[i].w; u = fa[u];
}
}
for ( int i = 1; i <= tot; ++ i)
for ( int k = 1; k <= tot; ++ k) check_Max(lv[i], a[i][k]);
for ( int i = 1; i < n; ++ i) aug(i);
int ans = 0;
for ( int i = 1; i <= tot; ++ i) ans += lv[i] + rv[i];
printf("%d\n", ans);
return 0;
}

[BZOJ 1937][Shoi2004]Mst 最小生成树的更多相关文章
- BZOJ 1937: [Shoi2004]Mst 最小生成树 [二分图最大权匹配]
传送门 题意: 给一张无向图和一棵生成树,改变一些边的权值使生成树为最小生成树,代价为改变权值和的绝对值,求最小代价 线性规划的形式: $Min\quad \sum\limits_{i=1}^{m} ...
- [BZOJ1937][SHOI2004]Mst最小生成树(KM算法,最大费用流)
1937: [Shoi2004]Mst 最小生成树 Time Limit: 3 Sec Memory Limit: 64 MBSubmit: 802 Solved: 344[Submit][Sta ...
- 【BZOJ1937】[Shoi2004]Mst 最小生成树 KM算法(线性规划)
[BZOJ1937][Shoi2004]Mst 最小生成树 Description Input 第一行为N.M,其中 表示顶点的数目, 表示边的数目.顶点的编号为1.2.3.…….N-1.N.接下来的 ...
- 【bzoj1937】 Shoi2004—Mst 最小生成树
http://www.lydsy.com/JudgeOnline/problem.php?id=1937 (题目链接) 题意 一个无向图,给出一个生成树,可以修改每条边的权值,问最小修改多少权值使得给 ...
- 【KM】BZOJ1937 [Shoi2004]Mst 最小生成树
这道题拖了好久因为懒,结果1A了,惊讶∑( 口 || [题目大意] 给定一张n个顶点m条边的有权无向图.现要修改各边边权,使得给出n-1条边是这张图的最小生成树,代价为变化量的绝对值.求最小代价之和. ...
- BZOJ1937 [Shoi2004]Mst 最小生成树
首先由贪心的想法知道,树边只减不加,非树边只加不减,令$w_i$表示i号边原来的边权,$d_i$表示i号边的改变量 对于一条非树边$j$连接着两个点$x$.$y$,则对于$xy$这条路径上的所有树边$ ...
- MST最小生成树
首先,贴上一个很好的讲解贴: http://www.wutianqi.com/?p=3012 HDOJ 1233 还是畅通工程 http://acm.hdu.edu.cn/showproblem.ph ...
- [poj1679]The Unique MST(最小生成树)
The Unique MST Time Limit: 1000MS Memory Limit: 10000K Total Submissions: 28207 Accepted: 10073 ...
- UVA 1151 Buy or Build (MST最小生成树,kruscal,变形)
题意: 要使n个点之间能够互通,要使两点直接互通需要耗费它们之间的欧几里得距离的平方大小的花费,这说明每两个点都可以使其互通.接着有q个套餐可以选,一旦选了这些套餐,他们所包含的点自动就连起来了,所需 ...
随机推荐
- Android代码速查,写给新手的朋友们[转]
原文地址:http://www.open-open.com/lib/view/open1397286499090.html 0 android 创建按钮 Button button = new But ...
- charles4.2下载与破解方法以及配置https.RP
Charles下载地址 地址:https://www.charlesproxy.com/latest-release/download.do 2. Charles破解 破解地址:https://www ...
- org.slf4j.impl.SimpleLoggerFactory cannot be cast to ch.qos.logback.classic.LoggerContext
查看日志信息: SLF4J: Class path contains multiple SLF4J bindings. SLF4J: Found binding in [jar:file:/app/a ...
- Linux-Linux基础入门
第一节 Linux系统简介 初步了解了什么是Linux系统,有何优势.与Windows系统有何不同,并了解Linux学习方法. 第二节 基本概念及概念 1.完成实验楼入门基础课程,共两个实验:(1)& ...
- C#中使用多线程访问Winform中控件的若干问题
我们在做winform应用的时候,大部分情况下都会碰到使用多线程控制界面上控件信息的问题.然而我们并不能用传统方法来做这个问题,下面我将详细的介绍. 首先来看传统方法: public partial ...
- 简述各大 Linux 发行版,有主观,不完全,望见谅
只罗列当前热门的linux发行版 更多关于 Linux 以及 Linux 衍生版的内容可以参阅 中文wiki Debian 系 Debian:开源社区的代表性 linux 系统,每2年一次更新,现在的 ...
- hdu 4284 Travel(壮压DP&TSP&floyd)
Travel Time Limit: 10000/5000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) Total Su ...
- XE10开发的APP对于苹果IPV6上架要求的处理(DelphiTeacher)
1.服务器必须使用域名.不能使用IP地址2.Indy的话,域名加[]3.DataSnap的话,Params.Values['CommunicationIPVersion'] :='IP_IPv6';4 ...
- C#基础入门 三
C#基础入门 三 类 类使用class关键字进行声明,前面加一个访问修饰符,public class car{} 访问修饰符:修师傅可以用来修饰类和类成员等,控制它们的可见度 修饰符关键字分别为:pu ...
- Tomcat应用配置
为Tomcat添加管理员 为了更好的管理tomcat服务器,我们通常会给tomcat添加用户管理员,这样就可以登录进入查看发布的项目.以下是实际操作步骤: 在Tomcat的配置目录下找到tomcat- ...