最小生成树的个数_最小生成树的实际应用

最小生成树的个数_最小生成树的实际应用给定一张 N 个点 M 条边的无向图,求无向图的严格次小生成树。设最小生成树的边权之和为 sum,严格次小生成树就是指边权之和大于 sum 的生成树中最小的一个。输入格式第一行包含两个整数 N 和 M。接下来 M 行,每行包含三个整数 x,y,z,表示点 x 和点 y 之前存在一条边,边的权值为 z。输出格式包含一行,仅一个数,表示严格次小生成树的边权和。(数据保证必定存在严格次小生成树)数据范围N≤105,M≤3×105输入样例:5 61 2 11 3 22 4 33 5 4

大家好,又见面了,我是你们的朋友全栈君。如果您正在找激活码,请点击查看最新教程,关注关注公众号 “全栈程序员社区” 获取激活教程,可能之前旧版本教程已经失效.最新Idea2022.1教程亲测有效,一键激活。

Jetbrains全系列IDE使用 1年只要46元 售后保障 童叟无欺

给定一张 N 个点 M 条边的无向图,求无向图的严格次小生成树。

设最小生成树的边权之和为 sum,严格次小生成树就是指边权之和大于 sum 的生成树中最小的一个。

输入格式
第一行包含两个整数 N 和 M。

接下来 M 行,每行包含三个整数 x,y,z,表示点 x 和点 y 之前存在一条边,边的权值为 z。

输出格式
包含一行,仅一个数,表示严格次小生成树的边权和。(数据保证必定存在严格次小生成树)

数据范围
N≤105,M≤3×105

输入样例:
5 6
1 2 1
1 3 2
2 4 3
3 5 4
3 4 3
4 5 6
输出样例:
11
#include<bits/stdc++.h>
using namespace std;
const int N = 1e5 + 10;
const int M = 3 * 3e5 + 10;
const int maxbit = 20;
typedef long long ll;
int f[N][maxbit],d1[N][maxbit],d2[N][maxbit],height[N];
struct Edge{ 
   
    int u,v,w,next;
    bool f;
    bool operator<(const Edge a)const{ 
   
        return w < a.w;
    }
}edge[M];
int head[N],cnt;
int n,m;
int fa[N];
int Find(int x){ 
   
    return fa[x] = (fa[x] == x ? x : Find(fa[x]));
}
void add(int u,int v,int w){ 
   
    edge[cnt].v = v;
    edge[cnt].w = w;
    edge[cnt].next = head[u];
    head[u] = cnt ++;
}
void dfs(int u,int fa,int d,int w){ 
   
    f[u][0] = fa;
    height[u] = d;
    d1[u][0] = w,d2[u][0] = 0;
    for(int i = 1;i <= maxbit - 1;i ++){ 
   
        f[u][i] = f[f[u][i - 1]][i - 1];
        d1[u][i] = max(d1[u][i - 1],d1[f[u][i - 1]][i - 1]);
        if(d1[u][i - 1] == d1[f[u][i - 1]][i - 1])
            d2[u][i] = max(d2[u][i - 1],d2[f[u][i - 1]][i - 1]);
        if(d1[u][i - 1] > d1[f[u][i - 1]][i - 1])
            d2[u][i] = max(d1[f[u][i - 1]][i - 1],d2[u][i - 1]);
        else d2[u][i] = max(d1[u][i - 1],d2[f[u][i - 1]][i - 1]);
    }
    for(int i = head[u];~i;i = edge[i].next){ 
   
        int v = edge[i].v,ww = edge[i].w;
        if(v == fa)continue;
        dfs(v,u,d + 1,ww);
    }
}
int lca(int x,int y,int &a,int &b){ 
   
    a = b = 0;
    if(height[x] < height[y])swap(x,y);
    for(int i = maxbit - 1;i >= 0;i --){ 
   
        if(height[f[x][i]] >= height[y]){ 
   
            if(d1[x][i] > a)b = a,a = d1[x][i];
            else if(d1[x][i] != a && d1[x][i] > b)b = d1[x][i]; 
            if(d2[x][i] > a)b = a,a = d2[x][i];
            else if(d2[x][i] != a && d2[x][i] > b)b = d2[x][i];
            x = f[x][i];
        }
    }
    if(x == y)return x;
    for(int i = maxbit - 1;i >= 0;i --){ 
   
        if(f[x][i] != f[y][i]){ 
   
            int aa = d1[x][i],bb = d2[x][i],cc = d1[y][i],dd = d2[y][i];
            if(aa > a)b = a,a = aa;
            else if(aa != a && aa > b)b = aa;
            if(bb > a)b = a,a = bb;
            else if(bb != a && bb > b)b = bb; 
            if(cc > a)b = a,a = cc;
            else if(cc != a && cc > b)b = cc; 
            if(dd > a)b = a,a = dd;
            else if(dd != a && dd > b)b = dd; 
            x = f[x][i],y = f[y][i];
        }
    }
    int aa = d1[x][0],bb = d2[x][0],cc = d1[y][0],dd = d2[y][0];
    if(aa > a)b = a,a = aa;
    else if(aa != a && aa > b)b = aa;
    if(bb > a)b = a,a = bb;
    else if(bb != a && bb > b)b = bb; 
    if(cc > a)b = a,a = cc;
    else if(cc != a && cc > b)b = cc; 
    if(dd > a)b = a,a = dd;
    else if(dd != a && dd > b)b = dd; 
    return f[x][0];
}
int main(){ 
   
    memset(head,-1,sizeof head);
    cin>>n>>m;
    int x,y,w;
    for(int i = 0;i < m;i ++){ 
   
        cin>>x>>y>>w;
        edge[i].u = x,edge[i].v = y,edge[i].w = w;
        edge[i].f = false;
    }
    cnt = m;
    sort(edge,edge + m);
    for(int i = 0;i <= n;i ++)fa[i] = i;
    ll res = 0;
    for(int i = 0;i < m;i ++){ 
   
        int a = Find(edge[i].u),b = Find(edge[i].v),w = edge[i].w;
        if(a != b){ 
   
            fa[a] = b;
            edge[i].f = true;
            add(edge[i].u,edge[i].v,w),add(edge[i].v,edge[i].u,w);
            res += w;
        }
    }
    
    dfs(1,0,1,0);
    int xx,yy;
    ll t = 0x3f3f3f3f3f3f3f3f;
    for(int i = 0;i < m;i ++){ 
   
        if(!edge[i].f){ 
   
            int a = edge[i].u,b = edge[i].v,w = edge[i].w;
            int aa = 0,bb = 0;
            lca(a,b,aa,bb);
            if(aa == w){ 
   
                t = min(res + w - bb,t);
            }
            else t = min(res + w - aa,t);
        }
    }
    cout<<t<<endl;
    
    return 0;
}
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/169114.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • pycharm2021.11.3激活补丁_最新在线免费激活

    (pycharm2021.11.3激活补丁)JetBrains旗下有多款编译器工具(如:IntelliJ、WebStorm、PyCharm等)在各编程领域几乎都占据了垄断地位。建立在开源IntelliJ平台之上,过去15年以来,JetBrains一直在不断发展和完善这个平台。这个平台可以针对您的开发工作流进行微调并且能够提供…

    2022年3月30日
    69
  • 在事务中执行批量复制操作SqlBulkCopy,SqlTransaction「建议收藏」

    在事务中执行批量复制操作SqlBulkCopy,SqlTransaction「建议收藏」    MicrosoftSQLServer包含名为bcp的常用命令行应用程序,用于快速将大文件批量复制到SQLServer数据库的表或视图中。使用SqlBulkCopy类可以编写提供类似功能的托管代码解决方案。还可以通过其他方式将数据加载到SQLServer表中(例如INSERT语句),但是SqlBulkCopy提供的性能要明显优于这些方式。   

    2022年5月8日
    153
  • div垂直居中的几种方式_div垂直水平居中

    div垂直居中的几种方式_div垂直水平居中利用CSS进行元素的水平居中,比较简单,行级元素设置其父元素的text-aligncenter,块级元素设置其本身的left和rightmargins为auto即可。本文收集了六种利用css进行元素的垂直居中的方法,每一种适用于不同的情况,在实际的使用过程中选择某一种方法即可。Line-HeightMethod试用:单行文本垂直居中,demo代码:

    2022年4月20日
    60
  • mac .bash_profile环境变量汇总

    mac .bash_profile环境变量汇总

    2021年12月5日
    60
  • 一个游戏是如何被设计和开发出来的[通俗易懂]

    一个游戏是如何被设计和开发出来的[通俗易懂]我在知乎回答“想要自己做一款游戏,需要学习哪些知识”下面简单列举了四个能力,分别是:程序、设计、美术、音乐。但是碍于篇幅限制,我并没有详细展开来说明每一项能力具体是如何发挥作用,以及发挥作用的形式和功效。如果在学习之前,我们对即将学习的东西一无所知的话,会导致学习中产生不小的迷茫感:不知道为何而学,不知道学了有什么作用,不知道该学习到什么程度。带着这样的迷茫去学习,会导致学习效率低下,容易受挫,甚…

    2022年5月27日
    35
  • 5G NR SRS (R15)[通俗易懂]

    5G NR SRS (R15)[通俗易懂]一、SRS序列对于SRS序列生成,其延续了LTE中采用的ZC序列,具体公式如下:[参考协议382115.2.2][参考协议382116.4.1.4.2]二、SRS时频资源1.时频资源NR中网络可以为终端配置一个或多个SRS资源集,多个资源集的目的可能是为了上下行多天线预编码,也有可能是为了上下行波束管理。一个SRS资源集内可以包含一个或多个SRS资源,每个SRS资源占…

    2025年8月20日
    3

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号