题目大意:

https://ac.nowcoder.com/acm/contest/272/D

在一个无向图中,给定一个起点,从起点开始走遍图中所有点

每条边有边权wi,表示第一次经过该道路时的花费(第二次及以后经过时花费为0)

此时用最少花费完成可能存在多种方案

求每种方案都必须经过的边有多少条

首先想到最小生成树

然后想到在得到最短边时 若存在其他长度相等的边 这条边此时就可被替代

但如果没有长度相等的边 那么这条边就是必须经过的边

然而这个想法经不起考验 是错误的 如下

但是没有长度相等的边 就是必须经过的边 这是毋庸置疑的

那么再观察下图

可以看到这是 一个只由长度相等的边组成的图

若在这个图中 我们要得到一棵生成树的话 边3-4是必选的 而剩下的1-2、2-3、3-1则任选两条即可

也就是说在生成树的所有选择方案里 这三条边不是必须经过的边

那么可以发现 在一个只由长度相等的边组成的图内 能形成一个环的几条边不是在所有的选择方案里必须经过的边

再结合kruskal得到最小生成树的步骤 每次先得到所有相同长度的最短边建图

再找到其中的环的个数m 那么要连接所有的环 必须经过的边就有m-1条

tarjan 求无向图内的环 就是在 有向图求强联通分量 的基础上进行修改

将 已走过的边 视为有向 不走其反向边

那么当走完这个图之后 整个图变成了一个有向图 此时图中的强联通分量就是环

如何 将已走过的边视为有向 呢

首先建图的过程中 对于一条边 我们是连了正向就连反向的 也就是这两条有向边在存储过程中的序号是连续的

所以我们从序号2开始存边的话 序号为 2和3 的两条有向边对应一条无向边 4和5对应一条无向边 6和7对应一条......

则对于 存储顺序为第 x 的有向边 其对应的反向边(即另一条有向边)顺序为 (x^1)

那么我们在递归时将上一条边的顺序 last 作为参数传过来 不走它对应的反向边即跳过顺序为 last^1 的边 就可以了

#include <bits/stdc++.h>
#define mem(i,j) memset(i,j,sizeof(i))
using namespace std;
const int N=2e5+; struct EDGE {
int u,v,w;
bool operator <(const EDGE& p)const {
return w<p.w;
}
}E[N<<];
struct NODE { int to,nt; }e[N<<];
int head[N], tot;
void addE(int u,int v) {
e[++tot].to=v;
e[tot].nt=head[u];
head[u]=tot;
}
int dfn[N], low[N], ind;
int fa[N], ans;
int n, m, p;
void init_e() {
ind=; mem(dfn,);
tot=; mem(head,);
}
void init_s() {
ans=;
for(int i=;i<=n;i++) fa[i]=i;
} int tarjan(int u,int last) {
dfn[u]=low[u]=++ind;
int res=;
for(int i=head[u];i;i=e[i].nt) {
if(i==(last^)) continue; // 是上一条边的对应反向边 跳过
int v=e[i].to;
if(!dfn[v]) {
res+=tarjan(v,i);
low[u]=min(low[u],low[v]);
} else low[u]=min(low[u],dfn[v]);
}
if(dfn[u]==low[u]) res++;
return res;
} int getfa(int x) {
if(x==fa[x]) return x;
return fa[x]=getfa(fa[x]);
}
void unite(int x,int y) {
x=getfa(x), y=getfa(y);
if(x!=y) fa[x]=y;
} void Kruskal() {
sort(E,E+m);
for(int i=,j;i<m;i=j) {
j=i;
while(j<m && E[j].w==E[i].w) j++; //找到所有与最短边相等的边
init_e(); // 初始化邻接表和tarjan需要的数组
for(int k=i;k<j;k++) { // 建图
int u=getfa(E[k].u), v=getfa(E[k].v);
if(u==v) continue; // 两个点已经连起来了
addE(u,v); addE(v,u);
}
for(int k=i;k<j;k++) {
int u=getfa(E[k].u);
if(!dfn[u]) ans+=tarjan(u,)-; //保证m个环连通 需要m-1条边
}
for(int k=i;k<j;k++)
unite(E[k].u,E[k].v);
}
} int main()
{
while(~scanf("%d%d%d",&n,&m,&p)) {
init_s();
for(int i=;i<m;i++)
scanf("%d%d%d",&E[i].u,&E[i].v,&E[i].w);
Kruskal();
printf("%d\n",ans);
} return ;
}

牛客D-Where are you /// kruskal+tarjan找无向图内的环的更多相关文章

  1. 牛客小白月赛12 I (tarjan求割边)

    题目链接:https://ac.nowcoder.com/acm/contest/392/I 题目大意:一个含有n个顶点m条边的图,求经过所有顶点必须要经过的边数. 例: 输入: 5 51 22 33 ...

  2. 牛客寒假算法基础集训营4 F(二分+拓扑判环)

    题目链接 题目的输出:对于每次提问,输出一行"Yes"表示大家都遵守了群规,反之输出"No". 那么输出的就是一连串的yes和no了,二分一下无环的最大提问位置 ...

  3. 牛客网 牛客练习赛7 D. 珂朵莉的无向图(多源BFS)

    题目链接  Problem D 比赛的时候完全想不到 直接对给定的这些点做多源$BFS$,把给定的这些点全都压到队列里,然后一个个做. 最后统计被访问的点的个数即可. #include <bit ...

  4. 【并查集缩点+tarjan无向图求桥】Where are you @牛客练习赛32 D

    目录 [并查集缩点+tarjan无向图求桥]Where are you @牛客练习赛32 D PROBLEM SOLUTION CODE [并查集缩点+tarjan无向图求桥]Where are yo ...

  5. 牛客网 牛客练习赛43 C.Tachibana Kanade Loves Review-最小生成树(并查集+Kruskal)+建虚点+读入挂

    链接:https://ac.nowcoder.com/acm/contest/548/C来源:牛客网 Tachibana Kanade Loves Review 时间限制:C/C++ 2秒,其他语言4 ...

  6. 牛客练习赛43 Tachibana Kanade Loves Review C(最小生成树Kruskal)

    链接:https://ac.nowcoder.com/acm/contest/548/C来源:牛客网 题目描述 立华奏是一个刚刚开始学习 OI 的萌新. 最近,实力强大的 QingyuQingyu 当 ...

  7. Applese 的毒气炸弹 G 牛客寒假算法基础集训营4(图论+最小生成树)

    链接:https://ac.nowcoder.com/acm/contest/330/G来源:牛客网 Applese 的毒气炸弹 时间限制:C/C++ 2秒,其他语言4秒 空间限制:C/C++ 262 ...

  8. 牛客练习赛32-D-MST+tarjin割边

    链接:https://ac.nowcoder.com/acm/contest/272/D来源:牛客网 题目描述 小p和他的朋友约定好去游乐场游玩,但是他们到了游乐场后却互相找不到对方了. 游乐场可以看 ...

  9. 牛客小白月赛12 I 华华和月月逛公园 (tarjian 求桥)

    链接:https://ac.nowcoder.com/acm/contest/392/I 来源:牛客网 华华和月月逛公园 时间限制:C/C++ 1秒,其他语言2秒 空间限制:C/C++ 32768K, ...

随机推荐

  1. windows10安装sqlmap与简单配置

    一.获取sqlmap压缩包: 注意:安装sqlmap之前,先确认已有python2.x环境 获取sqlmap压缩包并解压 下载地址:http://sqlmap.org/ 二.将解压缩的文件放在pyth ...

  2. 数据挖掘Aprior算法详解及c++源码

    [算法大致描述] Aprior算法主要有两个操作,扫描数据库+统计.计算每一阶频繁项集都要扫描一次数据库并且统计出满足支持度的n阶项集. [算法主要步骤] 一.频繁一项集 算法开始第一步,通过扫描数据 ...

  3. JMeter AI图片识别接口并发量测试

    由于临时接到一个性能测试任务,测试8个独立接口在实验室环境的TPS.响应时间以及服务器性能监控如CPU.内存.IO等,没有明确具体的响应时间与并发数,需求较模糊. 1.软件.硬件环境信息:JMeter ...

  4. HBase启动错误提示别的机器60000已经存在

    已经用cm  安装好了cdh,需要在上面添加HBase,并且做一个HBase故障转移功能,现在需要配置2个HMaster 在不同的机器上. 启动出现异常信息: 2015-12-23 14:44:38, ...

  5. 牛客练习赛48 D 小w的基站网络

    链接:https://ac.nowcoder.com/acm/contest/923/D来源:牛客网 时间限制:C/C++ 2秒,其他语言4秒 空间限制:C/C++ 262144K,其他语言52428 ...

  6. python基础讲解部分&纯小白需要扎实基础

    第一章知识点 一.Python简介 ​ python的创始人为吉多·范罗苏姆(Guido van Rossum),在中国人称龟叔 ​ Python崇尚优美.清晰.简单 应用领域: ​ (1)云计算,写 ...

  7. C语言之内存分配函数

    #include <stdio.h> #include <stdlib.h> #include <string.h> int main() { /********* ...

  8. 使用JMeter进行http压力测试

    一.背景及文档目的说明 采用JMeter测试工具对腾讯视频做负载测试,使用 JMeter图形结果和聚合图帮助测试系统在资源超负荷情况下的表现,以发现设计上的错误或验证系统的负载能力并估计系统瓶颈和并发 ...

  9. CyberArk

    CyberArk PIM 套件由5个部分组成: · CyberArk EPV (Enterprise Password Vault)– 企业密码保险库 基于CyberArk 专利的Vault技术,为企 ...

  10. mongodb java操作常用写法

    MongoDB 将数据存储为一个文档,数据结构由键值(key=>value)对组成.MongoDB 文档类似于 JSON 对象.字段值可以包含其他文档,数组及文档数组.下面介绍的是用java操作 ...