刷题总结——次小生成树(bzoj1977 最小生成树+倍增)
题目:
Description
小 C 最近学了很多最小生成树的算法,Prim 算法、Kurskal 算法、消圈算法等等。 正当小 C 洋洋得意之时,小 P 又来泼小 C 冷水了。小 P 说,让小 C 求出一个无向图的次小生成树,而且这个次小生成树还得是严格次小的,也就是说: 如果最小生成树选择的边集是 EM,严格次小生成树选择的边集是 ES,那么需要满足:(value(e) 表示边 e的权值)
这下小 C 蒙了,他找到了你,希望你帮他解决这个问题。
Input
第一行包含两个整数N 和M,表示无向图的点数与边数。 接下来 M行,每行 3个数x y z 表示,点 x 和点y之间有一条边,边的权值为z。
Output
包含一行,仅一个数,表示严格次小生成树的边权和。(数据保证必定存在严格次小生成树)
Sample Input
1 2 1
1 3 2
2 4 3
3 5 4
3 4 3
4 5 6
Sample Output
数据中无向图无自环; 50% 的数据N≤2 000 M≤3 000; 80% 的数据N≤50 000 M≤100 000; 100% 的数据N≤100 000 M≤300 000 ,边权值非负且不超过 10^9 。
题解:
次小生成树的模板题···
基本思路是求一次最小生成树··然后枚举那些非树边··找到以非树边两端点在树上路径中最大的一条边··将其删除并换上这条边然后计算目前答案··最后将所有算出的答案取min
唯一的问题是如何快速求得两端点原树路径上的最大边··可以采用树上倍增的方式··我们预处理出g[i][j],即i向上走2^j条边对应的祖先··以及maxx[i][j],走2^j的边中的最大值··
因为这道题是求严格次小的···我们不得不再预处理出一个minx[i][j],为次小值,至于如何求看代码吧··
然后就是常规的倍增思想··设我们枚举的非树边的两端点为a,b,我们找出它们在树上的lca,然后a和b在跳向lca时求得各自的最大值··注意由于是严格次大的··我们在跳的时候如果此时的maxx已经等于非树边的权值··我们就只能取minx···最后求ab中的最大值就是最大的一条边了
代码:
#include<iostream>
#include<cstdio>
#include<cstdlib>
#include<cmath>
#include<ctime>
#include<cstring>
#include<string>
#include<cctype>
#include<algorithm>
using namespace std;
const int N=1e5+;
const int M=3e5+;
struct node
{
int a,b,val;
}ed[M];
long long sum=;
int first[N],next[M*],go[M*],val[M*],tot,n,m;
int deep[N],maxx[N][],minx[N][],g[N][],father[N];
bool vis[M];
inline int R()
{
char c;int f=;
for(c=getchar();c<''||c>'';c=getchar());
for(;c<=''&&c>='';c=getchar()) f=(f<<)+(f<<)+c-'';
return f;
}
inline bool cmp(node a,node b)
{
return a.val<b.val;
}
inline int get(int a)
{
if(father[a]==a) return a;
else return father[a]=get(father[a]);
}
inline void comb(int a,int b,int c)
{
next[++tot]=first[a],first[a]=tot,go[tot]=b,val[tot]=c;
next[++tot]=first[b],first[b]=tot,go[tot]=a,val[tot]=c;
}
inline void kruscal()
{
sort(ed+,ed+m+,cmp);
int temp=;
for(int i=;i<=m;i++)
{
int fa=get(ed[i].a),fb=get(ed[i].b);
if(fa!=fb)
{
father[fa]=fb;temp++;sum+=ed[i].val;vis[i]=true;
comb(ed[i].a,ed[i].b,ed[i].val);
}
if(temp==n-) break;
}
}
inline void dfs(int u,int fa)
{
for(int e=first[u];e;e=next[e])
{
int v=go[e];if(v==fa) continue;
deep[v]=deep[u]+;maxx[v][]=val[e];g[v][]=u;dfs(v,u);
}
}
inline void pre()
{
for(int i=;i<=;i++)
for(int j=;j<=n;j++)
{
g[j][i]=g[g[j][i-]][i-];
maxx[j][i]=max(maxx[j][i-],maxx[g[j][i-]][i-]);
minx[j][i]=max(minx[j][i-],minx[g[j][i-]][i-]);
if(maxx[j][i-]<maxx[g[j][i-]][i-]&&minx[j][i]<maxx[j][i-]) minx[j][i]=maxx[j][i-];
else if(maxx[j][i-]>maxx[g[j][i-]][i-]&&minx[j][i]<maxx[g[j][i-]][i-]) minx[j][i]=maxx[g[j][i-]][i-];
}
}
inline int getlca(int a,int b)
{
if(deep[a]<deep[b]) swap(a,b);
int i,j;
for(i=;(<<i)<=deep[a];i++);i--;
for(j=i;j>=;j--)
if(deep[a]-(<<j)>=deep[b]) a=g[a][j];
if(a==b) return a;
for(i=;i>=;i--)
if(g[a][i]!=g[b][i]) a=g[a][i],b=g[b][i];
return g[a][];
}
inline int find(int a,int b,int lca,int lim)
{
int i,j,lmax=,rmax=;
for(i=;(<<i)<=deep[a];i++);i--;
for(j=i;j>=;j--)
if(deep[a]-(<<j)>=deep[lca])
{
if(maxx[a][j]!=lim) lmax=max(lmax,maxx[a][j]);
else lmax=max(lmax,minx[a][j]);
a=g[a][j];
}
for(i=;(<<i)<=deep[b];i++);i--;
for(j=i;j>=;j--)
if(deep[b]-(<<j)>=deep[lca])
{
if(maxx[b][j]!=lim) rmax=max(rmax,maxx[b][j]);
else rmax=max(rmax,minx[b][j]);
b=g[b][j];
}
return max(lmax,rmax);
}
inline void getans()
{
long long ans=2e+;
for(int i=;i<=m;i++)
if(!vis[i])
{
int a=ed[i].a,b=ed[i].b,c=ed[i].val;
int lca=getlca(a,b);
int temp=find(a,b,lca,c);
if(temp!=c&&ans>sum-temp+c) ans=sum-temp+c;
}
cout<<ans<<endl;
}
int main()
{
//freopen("a.in","r",stdin);
n=R(),m=R();
for(int i=;i<=n;i++) father[i]=i;
for(int i=;i<=m;i++) ed[i].a=R(),ed[i].b=R(),ed[i].val=R();
kruscal();
dfs(,);
pre();
getans();
return ;
}
刷题总结——次小生成树(bzoj1977 最小生成树+倍增)的更多相关文章
- 【BZOJ1977】[BeiJing2010组队]次小生成树 Tree 最小生成树+倍增
[BZOJ1977][BeiJing2010组队]次小生成树 Tree Description 小 C 最近学了很多最小生成树的算法,Prim 算法.Kurskal 算法.消圈算法等等. 正当小 C ...
- 【题解】洛谷P4180 [BJWC2010] 严格次小生成树(最小生成树+倍增求LCA)
洛谷P4180:https://www.luogu.org/problemnew/show/P4180 前言 这可以说是本蒟蒻打过最长的代码了 思路 先求出此图中的最小生成树 权值为tot 我们称这棵 ...
- P4180-[BJWC2010]严格次小生成树【Kruskal,倍增】
正题 题目链接:https://www.luogu.com.cn/problem/P4180 题目大意 \(n\)个点\(m\)条边的一张无向图,求它的严格次小生成树. \(1\leq n\leq 1 ...
- 【bzoj1977】[BeiJing2010组队]次小生成树 Tree 最小生成树+权值线段树合并
题目描述 求一张图的严格次小生成树的边权和,保证存在. 输入 第一行包含两个整数N 和M,表示无向图的点数与边数. 接下来 M行,每行 3个数x y z 表示,点 x 和点y之间有一条边,边的权值为z ...
- 严格次小生成树(Bzoj1977:[Beijing2010组队]次小生成树)
非严格次小生成树 很简单,先做最小生成树 然后枚举没加入的边加入,替换掉这个环内最大的边 最后取\(min\) 严格次小生成树 还是一样的 可以考虑维护一个严格次大值 最大值和枚举的边相同就替换次大值 ...
- 洛谷P4180 [BJWC2010]次小生成树(最小生成树,LCT,主席树,倍增LCA,倍增,树链剖分)
洛谷题目传送门 %%%TPLY巨佬和ysner巨佬%%% 他们的题解 思路分析 具体思路都在各位巨佬的题解中.这题做法挺多的,我就不对每个都详细讲了,泛泛而谈吧. 大多数算法都要用kruskal把最小 ...
- [bzoj1977][BeiJing2010组队]次小生成树 Tree——树上倍增+lca
Brief Description 求一个无向图的严格次小生成树. Algorithm Design 考察最小生成树的生成过程.对于一个非树边而言,如果我们使用这一条非树边去替换原MST的路径上的最大 ...
- (poj)1679 The Unique MST 求最小生成树是否唯一 (求次小生成树与最小生成树是否一样)
Description Given a connected undirected graph, tell if its minimum spanning tree is unique. Definit ...
- POJ 1679 The Unique MST (次小生成树 判断最小生成树是否唯一)
题目链接 Description Given a connected undirected graph, tell if its minimum spanning tree is unique. De ...
随机推荐
- Python实现购物小程序
一.需求 1.登录 { ‘xxx1’:{'passwd':'123','role':1,'moeny':10000,"carts":['mac']}, 'xxx1':{'passw ...
- 基于纹理内存的CUDA热传导模拟
原文链接 项目中有三个,第一个是全局内存,其余两个分别是基于1d和2d纹理内存.项目打包下载. 纹理内存是只读内存,与常量内存相同的是,纹理内存也缓存在芯片中,因此某些情况下,它能减少对内存的请求并提 ...
- 第17题:打印1到最大的n位数
面试题17:打印1到最大的n位数 题目:输入数字n,按顺序打印出从1最大的n位十进制数.比如输入3,则打印出1.2.3一直到最大的3位数即999. 考点: 用字符串或者数组表达一个大数. 思路 1. ...
- 1503: [NOI2004]郁闷的出纳员
Time Limit: 5 Sec Memory Limit: 64 MBSubmit: 13723 Solved: 4989[Submit][Status][Discuss] Descripti ...
- CF873B Balanced Substring (前缀和)
CF873B Balanced Substring (前缀和) 蛮有意思的一道题,不过还是.....................因为CF评测坏了,没有试过是否可过. 显然求\(\sum[i][0] ...
- centos下修改docker连接docker_host默认方式为tls方式
1.安装docker,请参考官网文档 centos下安装docker 2.安装完成应该可以使用docker的各种命令连接docker host.docker host运行在本机上,但与localhos ...
- keepalived原理(主从配置+haproxy)及配置文件详解
下图描述了使用keepalived+Haproxy主从配置来达到能够针对前段流量进行负载均衡到多台后端web1.web2.web3.img1.img2.但是由于haproxy会存在单点故障问题,因此使 ...
- tp5 使用paginate分页获取数据对象之后 如何对对象进行数据添加
tp5 使用paginate分页获取数据对象之后 如何对对象进行数据添加 大家都知道,在使用tp5的paginate获取分页数据之后,得到的是一个数据对象,但有时会碰到要对数据对象进行二次加工的情况, ...
- day 44 前端HTML
前端HTML HTML介绍 Web服务本质 import socket sk = socket.socket() sk.bind(("127.0.0.1", 8080)) sk ...
- Java Web 基础(一) 基于TCP的Socket网络编程
一.Socket简单介绍 Socket通信作为Java网络通讯的基础内容,集中了异常.I/O流模式等众多知识点.学习Socket通信,既能够了解真正的网络通讯原理,也能够增强对I/O流模式的理解. 1 ...