【XSY2485】MST(最小生成树+倍增lca+并查集)

题面

Description

给定一个\(n\)个点\(m\)条边的连通图,保证没有自环和重边。对于每条边求出,在其他边权值不变的情况下,它能取的最大权值,使得这条边在连通图的所有最小生成树上。假如最大权值为无限大,则输出\(-1\)。

Input

第一行两个整数\(n\),\(m\),表示\(n\)个点\(m\)条边

接下来\(m\)行,每行\(3\)个整数\(x\),\(y\),\(z\),表示节点\(x\)和节点\(y\)之间有一条长\(z\)的边。

Output

输出一行\(m\)个整数,表示每条边的答案

Sample Input

4 4
1 2 2
2 3 2
3 4 2
4 1 3

Sample Output

2 2 2 1 

HINT

对于\(30\%\)的数据\(1≤n≤10^3\),\(1≤m≤3\times10^3\)

对于\(100\%\)的数据\(1≤n,m≤2\times 10^5\),\(1≤z≤10^9\)

题解

(来自XSY的solution)

先求出图的一棵最小生成树:

对于不在树上的边\((x,y)\), 它的权值只要小于树上\(x\)到\(y\)路径中一条边就可以代替这条边。

对于在树上的边\((x,y)\),可以先预处理出所有两端在\(x\)到\(y\)路径上的不在树上的边的最小值。它的权值一定要小于最小值。

路径\(max\)和\(min\)都可以用倍增求。

时间复杂度\(O(nlogn)\)。

代码:

#include<bits/stdc++.h>

#define N 200010
#define M127 2139062143

using namespace std;

struct edge
{
    int u,v,w,id;
}e[N];

int n,m,fa[N],ans[N];
int cnt,head[N],to[N<<1],nxt[N<<1],w[N<<1],id[N<<1];
int f[N][20],maxn[N][20],from[N],d[N];
bool flag[N];

void adde(int u,int v,int wi,int idi)
{
    to[++cnt]=v;
    w[cnt]=wi;
    id[cnt]=idi;
    nxt[cnt]=head[u];
    head[u]=cnt;
}

bool cmp(edge a,edge b)
{
    return a.w<b.w;
}

int find(int x)
{
    if(fa[x]!=x)
        return fa[x]=find(fa[x]);
    return x;
}

void dfs(int u)
{
    for(int i=1;i<=18;i++)
    {
        f[u][i]=f[f[u][i-1]][i-1];
        maxn[u][i]=max(maxn[u][i-1],maxn[f[u][i-1]][i-1]);
    }
    for(int i=head[u];i;i=nxt[i])
    {
        if(to[i]!=f[u][0])
        {
            f[to[i]][0]=u;
            maxn[to[i]][0]=w[i];
            from[to[i]]=id[i];
            d[to[i]]=d[u]+1;
            dfs(to[i]);
        }
    }
}

int getMax(int a,int b,int &lca)
{
    int ans=0;
    if(d[a]<d[b])
        swap(a,b);
    for(int i=18;i>=0;i--)
    {
        if(d[f[a][i]]>=d[b])
        {
            ans=max(ans,maxn[a][i]);
            a=f[a][i];
        }
    }
    if(a==b)
    {
        lca=a;
        return ans;
    }
    for(int i=18;i>=0;i--)
    {
        if(f[a][i]!=f[b][i])
        {
            ans=max(ans,maxn[a][i]);
            ans=max(ans,maxn[b][i]);
            a=f[a][i],b=f[b][i];
        }
    }
    lca=f[a][0];
    return max(ans,max(maxn[a][0],maxn[b][0]));
}

void solve(int u,int lca,int wi)
{
    u=find(u);
    while(d[u]>d[lca])
    {
        ans[from[u]]=min(ans[from[u]],wi-1);
        fa[u]=find(f[u][0]);
        u=find(u);
    }
}

int main()
{
    memset(ans,127,sizeof(ans));
    scanf("%d%d",&n,&m);
    for(int i=1;i<=n;i++)
        fa[i]=i;
    for(int i=1;i<=m;i++)
    {
        scanf("%d%d%d",&e[i].u,&e[i].v,&e[i].w);
        e[i].id=i;
    }
    sort(e+1,e+m+1,cmp);
    for(int i=1,tot=0;i<=m;i++)
    {
        if(tot==n-1)
            break;
        int x=find(e[i].u);
        int y=find(e[i].v);
        if(x!=y)
        {
            fa[y]=x;
            adde(e[i].u,e[i].v,e[i].w,e[i].id);
            adde(e[i].v,e[i].u,e[i].w,e[i].id);
            flag[i]=true;
            tot++;
        }
    }
    dfs(1);
    for(int i=1;i<=n;i++)
        fa[i]=i;
    for(int i=1;i<=m;i++)
    {
        if(!flag[i])
        {
            int lca;
            ans[e[i].id]=getMax(e[i].u,e[i].v,lca)-1;
            solve(e[i].u,lca,e[i].w);
            solve(e[i].v,lca,e[i].w);
        }
    }
    for(int i=1;i<=m;i++)
    {
        if(ans[i]==M127)printf("-1 ");
        else printf("%d ",ans[i]);
    }
    return 0;
}

原文地址:https://www.cnblogs.com/ez-lcw/p/11520267.html

时间: 2024-10-09 04:20:51

【XSY2485】MST(最小生成树+倍增lca+并查集)的相关文章

【CodeForces】827 D. Best Edge Weight 最小生成树+倍增LCA+并查集

[题意]给定n个点m条边的带边权无向连通图,对每条边求最大边权,满足其他边权不变的前提下图的任意最小生成树都经过它.n,m<=2*10^5,1<=wi<=10^9. [算法]最小生成树+倍增LCA+并查集 [题解]首先求出图的一个最小生成树,则所有边分成树边和非树边. 对于非树边(u,v),假设u和v在最小生成树上的路径的最大边权Max,那么一定满足w(u,v)<=Max /////////////////////////////////////// 原文地址:https://ww

【BZOJ-3910】火车 倍增LCA + 并查集

3910: 火车 Time Limit: 20 Sec  Memory Limit: 512 MBSubmit: 262  Solved: 90[Submit][Status][Discuss] Description A 国有n 个城市,城市之间有一些双向道路相连,并且城市两两之间有唯一路径.现在有火车在城市 a,需要经过m 个城市.火车按照以下规则行驶:每次行驶到还没有经过的城市中在 m 个城市中最靠前的.现在小 A 想知道火车经过这m 个城市后所经过的道路数量. Input 第一行三个整数

【bzoj4242】水壶 BFS+最小生成树+倍增LCA

题目描述 JOI君所居住的IOI市以一年四季都十分炎热著称. IOI市是一个被分成纵H*横W块区域的长方形,每个区域都是建筑物.原野.墙壁之一.建筑物的区域有P个,编号为1...P. JOI君只能进入建筑物与原野,而且每次只能走到相邻的区域中,且不能移动到市外. JOI君因为各种各样的事情,必须在各个建筑物之间往返.虽然建筑物中的冷气设备非常好,但原野上的日光十分强烈,因此在原野上每走过一个区域都需要1单位的水.此外,原野上没有诸如自动售货机.饮水处之类的东西,因此IOI市的市民一般都携带水壶出

训练指南 UVA - 11354(最小生成树 + 倍增LCA)

layout: post title: 训练指南 UVA - 11354(最小生成树 + 倍增LCA) author: "luowentaoaa" catalog: true mathjax: true tags: - 最小生成树 - LCA - 图论 - 训练指南 Bond UVA - 11354 题意 给你一张无向图,然后有若干组询问,让你输出a->b的最小瓶颈路 题解 先求出最小生成树,然后对这个最小生成树做LCA. #include<bits/stdc++.h>

Hdu 5458 Stability (LCA + 并查集 + 树状数组 + 缩点)

题目链接: Hdu 5458 Stability 题目描述: 给出一个还有环和重边的图G,对图G有两种操作: 1 u v, 删除u与v之间的一天边 (保证这个边一定存在) 2 u v, 查询u到v的路径上有几条桥. 解题思路: 这个题目有很多次操作,包含查询和删边两类,首先想到的是连通分量加缩点.如果按照顺序来,删边时候求桥就是问题了.所以可以离线处理,然后一边记录答案一边加边缩点. 对于一个图,把连通分量缩成一个点后,这个图就成为了一棵树, 然后深度差就等于桥的数目.查询的时候对于(u, v)

HDU 5458 Stability(双连通分量+LCA+并查集+树状数组)(2015 ACM/ICPC Asia Regional Shenyang Online)

题目大意:给一个N个点M条边的无向图,有Q个询问:1.删掉a.b之间所存在的边:2.询问有多少条边,单独删掉之后a与b不再连通. 思路:脑洞大开. 对于询问,首先想到的就是a与b之间有多少桥(割边),然后想到双连通分量,然而删边是个坑爹的问题,于是我们离线倒着来,把删边变成加边. 双连通分量这种东西呢,其实缩点连起来之后,就是一棵树辣. 然后询问两个点的时候,设根到点x的距离为dep[x],a.b的最近公共祖先为lca(a, b),那么询问query(a, b) = dep[a] + dep[b

【BZOJ3910】火车 LCA+并查集

链接: #include <stdio.h> int main() { puts("转载请注明出处[vmurder]谢谢"); puts("网址:blog.csdn.net/vmurder/article/details/44780959"); } 题解: 首先找两点之间路径可以用倍增LCA. 然后标记哪个点走过可以用并查集,均摊下来最后是线性的. 代码: #include <cstdio> #include <cstring>

BZOJ 3910 火车 LCA+并查集

题目大意 给出一棵树,起点,和要经过的点的序列,已经经过的点就不用去了,剩下的点按照顺序依次去,问要经过多少条边. 思路 链剖大概应该是可以,不过没试,用了听大爷说的一种神奇的方法. 因为树上经过的点肯定是一段一段的,就想到用并查集将一段合成一个点,每个点最多只能被合一次,这样的话就能保证时间复杂度.查询的时候像链剖一样一段一段往上跳就行了,还要顺便把路径上的所有点缩起来. CODE #define _CRT_SECURE_NO_WARNINGS #include <cstdio> #incl

最小生成树Kruskal算法+并查集实现

今天刚掌握Kruskal算法,写下随笔. 对于稀疏图来说,用Kruskal写最小生成树效率更好,加上并查集,可对其进行优化. Kruskal算法的步骤: 1.对所有边进行从小到大的排序. 2.每次选一条边(最小的边),如果如果形成环,就不加入(u,v)中,否则加入.那么加入的(u,v)一定是最佳的. 并查集: 我们可以把每个连通分量看成一个集合,该集合包含了连通分量的所有点.而具体的连通方式无关紧要,好比集合中的元素没有先后顺序之分,只有"属于"与"不属于"的区别.