JZOJ 4313. 【NOIP2015模拟11.4】电话线铺设
题目



思路
先求只用王牌电缆的最小生成树
再选一条李牌电缆替换王牌电缆
使答案最小就完了
假如要替换的李牌电缆两端点是 \(u,v\)
那么生成树中 \(u \Longrightarrow lca(u,v)\) 和 \(v \Longrightarrow lca(u,v)\) 这两条链中的权值最大的边就是要替换的边
类似于次小生成树
倍增维护就好了
注意,有可能只用王牌电缆无法构成最小生成树
这里特判一下,此时跑最小生成树最终的结果必然是两个不连通的集合
这使枚举的李牌电缆就要使它们联通,求一个最小的李牌电缆即可
#include<cstdio>
#include<algorithm>
using namespace std;
const int N = 2e5 + 5;
int n , W , L , h[N] , fa[N] , tot , vis[N] , size[N] , lw , lid , hl;
int f[N][20] , anc[N][20] , ind[N][20] , dep[N] , ans , s , rt;
struct edge{
int nxt , to , w , id;
}e[N << 1];
struct Edge{
int u , v , w , id;
}E[N];
struct Edge1{
int u , v , w;
}l[N];
inline void addedge(int u , int v , int w , int id)
{
e[++tot] = (edge){h[u] , v , w , id};
h[u] = tot;
}
inline bool cmpE(Edge x , Edge y){return x.w < y.w;}
inline int find(int x){return fa[x] == x ? x : fa[x] = find(fa[x]);}
inline bool merge(int x , int y)
{
int xx = find(x) , yy = find(y);
if (fa[xx] != fa[yy])
{
if (size[xx] < size[yy]) fa[xx] = fa[yy] , size[yy] += size[xx];
else fa[yy] = fa[xx] , size[xx] += size[yy];
return 1;
}
return 0;
}
inline void Kruskal()
{
int u , v , w;
for(register int i = 1; i <= W; i++)
scanf("%d%d%d" , &E[i].u , &E[i].v , &E[i].w) , E[i].id = i;
sort(E + 1 , E + W + 1 , cmpE);
for(register int i = 1; i <= n; i++) fa[i] = i , size[i] = 1;
for(register int i = 1; i <= W; i++)
{
if (merge(E[i].u , E[i].v))
{
ans += E[i].w , s++;
vis[E[i].id] = 1;
addedge(E[i].u , E[i].v , E[i].w , E[i].id);
addedge(E[i].v , E[i].u , E[i].w , E[i].id);
if (!rt) rt = E[i].u;
}
if (s == n - 1) break;
}
}
inline void dfs(int x , int father)
{
for(register int i = 1; i <= 17; i++)
if (f[x][i - 1])
{
f[x][i] = f[f[x][i - 1]][i - 1];
if (anc[x][i - 1] < anc[f[x][i - 1]][i - 1])
anc[x][i] = anc[f[x][i - 1]][i - 1] , ind[x][i] = ind[f[x][i - 1]][i - 1];
else anc[x][i] = anc[x][i - 1] , ind[x][i] = ind[x][i - 1];
}
else break;
for(register int i = h[x]; i; i = e[i].nxt)
{
int v = e[i].to;
if (v == father) continue;
f[v][0] = x , dep[v] = dep[x] + 1 , anc[v][0] = e[i].w , ind[v][0] = e[i].id;
dfs(v , x);
}
}
inline void update(int ll , int u , int v)
{
if (dep[u] < dep[v]) swap(u , v);
int deep = dep[u] - dep[v] , sum;
int t = -0x3f3f3f3f , id;
for(register int i = 0; i <= 17; i++)
if (deep & (1 << i))
{
if (t < anc[u][i]) t = anc[u][i] , id = ind[u][i];
u = f[u][i];
}
if (u == v)
{
sum = ans - t + l[ll].w;
if (sum < lw) lw = sum , lid = ll , hl = id;
return;
}
for(register int i = 17; i >= 0; i--)
if (f[u][i] != f[v][i])
{
if (t < anc[u][i]) t = anc[u][i] , id = ind[u][i];
if (t < anc[v][i]) t = anc[v][i] , id = ind[v][i];
u = f[u][i] , v = f[v][i];
}
if (t < anc[u][0]) t = anc[u][0] , id = ind[u][0];
if (t < anc[v][0]) t = anc[v][0] , id = ind[v][0];
sum = ans - t + l[ll].w;
if (sum < lw) lw = sum , lid = ll , hl = id;
}
inline void getans()
{
for(register int i = 1; i <= L; i++) scanf("%d%d%d" , &l[i].u , &l[i].v , &l[i].w);
if (s != n - 1)
{
int Min = 0x3f3f3f3f , ld = 0;
for(register int i = 1; i <= L; i++)
if (find(l[i].u) != find(l[i].v) && l[i].w < Min)
Min = l[i].w , ld = i;
printf("%d\n" , ans + Min);
for(register int i = 1; i <= W; i++)
if (vis[i]) printf("%d\n" , i);
printf("%d\n" , ld);
return;
}
dfs(rt , 0);
lw = 0x3f3f3f3f;
for(register int i = 1; i <= L; i++) update(i , l[i].u , l[i].v);
printf("%d\n" , lw) , vis[hl] = 0;
for(register int i = 1; i <= W; i++)
if (vis[i]) printf("%d\n" , i);
printf("%d\n" , lid);
}
int main()
{
freopen("telephone.in" , "r" , stdin);
freopen("telephone.out" , "w" , stdout);
scanf("%d%d%d" , &n , &W , &L);
Kruskal();
getans();
}
JZOJ 4313. 【NOIP2015模拟11.4】电话线铺设的更多相关文章
- [JZOJ 4307] [NOIP2015模拟11.3晚] 喝喝喝 解题报告
题目链接: http://172.16.0.132/senior/#main/show/4307 题目: 解题报告: 题目询问我们没出现坏对的连续区间个数 我们考虑从左到有枚举右端点$r$,判断$a[ ...
- JZOJ 4298. 【NOIP2015模拟11.2晚】我的天
4298. [NOIP2015模拟11.2晚]我的天 (File IO): input:ohmygod.in output:ohmygod.out Time Limits: 1000 ms Memor ...
- 【NOIP2015模拟11.5】JZOJ8月5日提高组T2 Lucas的数列
[NOIP2015模拟11.5]JZOJ8月5日提高组T2 Lucas的数列 题目 PS:\(n*n*T*T<=10^{18}\)而不是\(10^1*8\) 题解 题意: 给出\(n\)个元素的 ...
- 【NOIP2015模拟11.5】JZOJ8月5日提高组T1 俄罗斯套娃
[NOIP2015模拟11.5]JZOJ8月5日提高组T1 俄罗斯套娃 题目 题解 题意就是说 将1~\(n\)排列,问有多少种方案使得序列的逆序对个数小于\(k\) 很容易想到DP 设\(f[i][ ...
- 【NOIP2015模拟11.2晚】JZOJ8月4日提高组T2 我的天
[NOIP2015模拟11.2晚]JZOJ8月4日提高组T2 我的天 题目 很久很以前,有一个古老的村庄--xiba村,村子里生活着n+1个村民,但由于历届村长恐怖而且黑暗的魔法统治下,村民们各自过着 ...
- 【NOIP2015模拟11.4】JZOJ2020年8月6日提高组T2 最优交换
[NOIP2015模拟11.4]JZOJ2020年8月6日提高组T2 最优交换 题目 题解 题意 有一个长度为\(n\)的正整数 最多可以进行\(k\)次操作 每次操作交换相邻两个位置上的数 问可以得 ...
- 【NOIP2015模拟11.4】JZOJ8月6日提高组T1 刷题计划
[NOIP2015模拟11.4]JZOJ8月6日提高组T1 刷题计划 题目 题解 题意 有\(n\)道题,编号为1~\(n\) 给出\(m\)次操作 每次操作有3种类型 1 \(x\) 表示交了\(A ...
- 【NOIP2015模拟11.5】JZOJ8月5日提高组T3 旅行
[NOIP2015模拟11.5]JZOJ8月5日提高组T3 旅行 题目 若不存在第\(k\)短路径时,输出"Stupid Mike" 题解 题意 给出一个有\(n\)个点的树 问这 ...
- 【NOIP2015模拟11.3】备用钥匙
题目 你知道Just Odd Inventions社吗?这个公司的业务是"只不过是奇妙的发明(Just Odd Inventions)".这里简称为JOI社. JOI社有N名员工, ...
- JZOJ4307. 【NOIP2015模拟11.3晚】喝喝喝
Description
随机推荐
- 【极客时间】大数据概述及HDFS介绍
- 深刻理解JAVA并发中的有序性问题和解决之道
欢迎关注专栏[JAVA并发] 更多技术干活尽在个人公众号--JAVA旭阳 问题 Java并发情况下总是会遇到各种意向不到的问题,比如下面的代码: int num = 0; boolean ready ...
- MongoDB海量数据分页查询优化
MongoDB海量数据分页查询优化 一.背景 大量数据需从Mongo拿出来,一次性拿出来不科学,传统分页效率低下 二.传统方案 就是最常规的方案,假设 我们需要对文章 articles 这个表(集合) ...
- 解决PC 拖动浏览器或者应用时CPU占用过高问题
电脑配置5800X.6800XT.win11在用edge或chrome只打开百度和博客园首页的情况下,长按来回拖动浏览器窗口,会发现CPU占用直线上升.能直接从4%变成70-80%导致系统卡住...我 ...
- python模块的含义
目录 模块简介 模块的本质 python模块的历史 python模块的表现形式 模块的分类 导入模块的两种句式 强调 import句式 import流程推导 练习 from...import...句式 ...
- Burp Suite安装
1.Burpsuite简介 Burp Suite是一款集成化的渗透测试工具,包含了很多功能,可以帮助我们高效地完成对Web应用程序的渗透测试和攻击. Burp Suite 由Java语言编写,基于J ...
- 开发一个MyBatis通用Mapper的轮子
一.前言 程序猿为什么如此执着于造轮子?MyBatis-Plus如此强大的工具流行这么多年了,我为啥还在重复造这样的轮子? 1.公司的技术规范不允许使用MyBatis-Plus,咱也不知道什么原因: ...
- 前端h5适配刘海屏和滴水屏
前端适配苹果刘海屏,安卓刘海屏水滴瓶 其实w3c早就为我们提供了解决方法(CSS3新特性viewport-fit) 在w3c.org官方给出的关于圆形展示(Round display)的标准中, 提到 ...
- 【转载】【WinAPI】LockWindowUpdate的函数的用法
DelPhi LockWindowUpdate的函数的用法 Application.ProcessMessages; LockWindowUpdate(Self.Handle); //锁住当前窗口 L ...
- Java中Elasticsearch 实现分页方式(三种方式)
目录 ES 简介 ES 的特点: 一.from + size 浅分页 二.scroll 深分页 scroll删除 三.search_after 深分页 ES 简介 Elasticsearch 是一个基 ...