【牛客 - 272D】Where are you(Tarjan求桥)

题干:
 

小p和他的朋友约定好去游乐场游玩,但是他们到了游乐场后却互相找不到对方了。
游乐场可以看做是一张n个点,m条道路的图,每条道路有边权wi,表示第一次经过该道路时的花费(第二次及以后经过时花费为0)。
现在,小p要去找他的朋友,但他的朋友行踪很诡异,小p总是要遍历完这n个点才能找到他,同时小p希望总花费最小。
找到朋友的方案可能不唯一(具体看样例解释),小p想知道在这所有的方案中,有多少条边在每个方案中都会被经过。
 

输入描述:

第一行两个整数n, m. p,分别表示点数,边数,小p的初始位置。
接下来m行,每行两个整数u, v, w表示从u到v有一条无向边,边权为w。

输出描述:

输出一个整数k,表示必须经过的边的数量。

示例1

输入

复制

5 7 1
1 2 3
2 3 7
1 3 5
2 4 2
1 5 3
5 4 3
2 5 3

输出

复制

2

说明

 

 

样例解释:

 


几种可能的方案如下:
1−2−4−5−4−2−1−31−2−4−5−4−2−1−3
1−5−4−2−4−5−1−31−5−4−2−4−5−1−3
1−3−1−2−5−2−41−3−1−2−5−2−4
1−2−5−2−4−2−5−2−4−2−5−2−4⋯−2−5−2−4−2−1−31−2−5−2−4−2−5−2−4−2−5−2−4⋯−2−5−2−4−2−1−3
可以证明,4 - 2和1 - 3这两条边在所有方案中都被经过。
(以上每种方案的总花费均为13,同时可以证明没有比这更优的策略)

示例2

输入

复制

3 3 1
1 2 1
1 3 1
2 3 2

输出

复制

2

示例3

输入

复制

3 3 1
1 2 2
2 3 2
1 3 2

输出

复制

0

备注:

 

2⩽n<m⩽2∗105,1⩽边权⩽1062⩽n<m⩽2∗105,1⩽边权⩽106

保证图联通,保证无自环,保证无重边

解题报告:

  好难的题,,不会证明但是倒是想明白了AC代码2那里为什么else if(i!=(lstedge^1)) low[x]=min(low[x],dfn[y]);这里不能写成 low[x]=min(low[x],low[y]);因为你想啊(虽然这个题能过)。

按照这个边的顺序访问节点,那么当访问边7的时候,LOW[3]已经变成了2,所以如果那样写,那LOW[6]也变成了2.看起来是没什么问题,但是我们看回溯到3这个节点的时候(注意这时候还没访问3->6这条边,刚刚那是6节点开始的6->3这条边),他需要开始看能否把4节点这一支分出去当成一个子图了,发现不行,因为DFN[4]=2(这本是不对的,但是托了3->2这条边优先访问的福啊!就把状态弄成这样了)然后再遍历3->6这个节点,发现也不行,因为DFN[6]=2,然后无奈返回,并且不标记成是个割点,,但是很显然,3这个点就是个割点!!!这就是为什么板子要这么写!!

对于题目:

图片说明

图片说明

AC代码: 

#include<bits/stdc++.h>
using namespace std;
const int maxn=200050;
int root,sum=1,dfn[maxn],low[maxn],ans[maxn],f[maxn];
int getf(int u){return f[u]==u?u:f[u]=getf(f[u]);}
struct Ed{int u,v,w;bool operator<(const Ed &A)const{return w<A.w;
}
}E[maxn];
struct Edge{int v,id,next;}e[maxn*2];
int first[maxn],tot;
void add(int u,int v,int id){e[tot].v=v;e[tot].id=id;e[tot].next=first[u];first[u]=tot++;
}
void tarjin(int u,int last){dfn[u]=low[u]=sum++;for(int i=first[u];~i;i=e[i].next){int v=e[i].v;if(i==(1^last))continue;if(dfn[v]){low[u]=min(low[u],dfn[v]);}else{tarjin(v,i);low[u]=min(low[u],low[v]);if(low[v]>dfn[u])ans[e[i].id]=1;}}
}
int main(){int i,j,n,m,p,u,v,w;scanf("%d%d%*d",&n,&m);tot=0;memset(first,-1,sizeof(first));for(i=1;i<=n;++i)f[i]=i;for(i=0;i<m;++i){scanf("%d%d%d",&E[i].u,&E[i].v,&E[i].w);}sort(E,E+m);for(i=0;i<m;++i){j=i;while(j+1<m&&E[j+1].w==E[i].w)j++;tot=0;for(int k=i;k<=j;++k){int fu=getf(E[k].u),fv=getf(E[k].v);if(fu!=fv){add(fu,fv,k);add(fv,fu,k);}}for(int k=i;k<=j;++k){int fu=getf(E[k].u),fv=getf(E[k].v);if(fu==fv || dfn[fu]) continue;tarjin(fu,-1);}for(int k=i;k<=j;++k){int fu=getf(E[k].u),fv=getf(E[k].v);if(fu==fv)continue;first[fu]=first[fv]=-1;dfn[fu]=dfn[fv]=0;f[fu]=fv;}i=j;}int h=0;for(i=0;i<m;++i)h+=ans[i];cout<<h<<endl;return 0;
}

AC代码2:

#include<bits/stdc++.h>
using namespace std;
#define maxn 200001
#define maxm 300001
struct edge
{int a,b,c;int id;inline void read(int _id){scanf("%d%d%d",&a,&b,&c);id=_id;}bool operator <(const edge &p)const{return c<p.c;}
}e[maxm];
int n,m;
inline void init()
{int p;scanf("%d%d",&n,&m);scanf("%d",&p);for(int i=1;i<=m;i++) e[i].read(i);sort(e+1,e+m+1);
}
int head[maxn],nxt[maxm<<1],ver[maxm<<1],id[maxm<<1],tot;
inline void addedge(int a,int b,int _id)
{nxt[++tot]=head[a];ver[tot]=b;id[tot]=_id;head[a]=tot;nxt[++tot]=head[b];ver[tot]=a;id[tot]=_id;head[b]=tot;
}
int ans[maxm];int dfn[maxn],low[maxn],cnt;
inline void tarjan(int x,int lstedge)
{dfn[x]=low[x]=++cnt;for(int i=head[x];i;i=nxt[i]){int y=ver[i];if(!dfn[y]){tarjan(y,i);low[x]=min(low[x],low[y]);if(low[y]>dfn[x]) ans[id[i]]=666;}else if(i!=(lstedge^1)) low[x]=min(low[x],dfn[y]);}
}
int fa[maxn];inline int find(int x){return fa[x]==x?x:fa[x]=find(fa[x]);}
int main()
{init();for(int i=1;i<=n;i++) fa[i]=i;int Nowedge=1;while(Nowedge<=m){int L=Nowedge,R=Nowedge;while(R+1<=m&&e[R].c==e[R+1].c) R++;Nowedge=R+1;cnt=0;tot=1;for(int i=L;i<=R;i++){int a=find(e[i].a),b=find(e[i].b);head[a]=0;head[b]=0;dfn[a]=dfn[b]=low[a]=low[b]=0;}for(int i=L;i<=R;i++){int a=find(e[i].a),b=find(e[i].b);if(a==b){ans[e[i].id]=-1;continue;}ans[e[i].id]=233;addedge(a,b,e[i].id);}for(int i=L;i<=R;i++){if(!dfn[find(e[i].a)]) tarjan(find(e[i].a),0);if(!dfn[find(e[i].b)]) tarjan(find(e[i].b),0);}for(int i=L;i<=R;i++) fa[find(e[i].a)]=find(e[i].b);}int ttl = 0;for(int i=1;i<=m;i++){if(ans[i]==666) ttl++;}cout<<ttl<<endl;return 0;
}

AC代码3:

#include<bits/stdc++.h>
using namespace std;
const int N=2e5+10;
struct edge{int f,to,v;}a[N];
struct node{int to,id;};
bool cmp(edge a,edge b){return a.v<b.v;}
int fa[N];vector<node>g[N];
int dfn[N],low[N],num;int ans[N];
int find(int x){return fa[x]==x?x:fa[x]=find(fa[x]);}
void unite(int x,int y){x=find(x),y=find(y),fa[x]=y;}
void tarjan(int x,int fa){dfn[x]=low[x]=++num;for(auto it:g[x]){int u=it.to,id=it.id;if(id==fa) continue;if(!dfn[u]){tarjan(u,id);low[x]=min(low[x],low[u]);if(dfn[x]<low[u]) ans[id]=1;}else low[x]=min(low[x],dfn[u]);}
}
int main(){int n,m,q,x,y,c;scanf("%d%d%d",&n,&m,&q);for(int i=1;i<=m;i++) scanf("%d%d%d",&a[i].f,&a[i].to,&a[i].v);sort(a+1,a+m+1,cmp);for(int i=1;i<=n;i++) fa[i]=i;for(int i=1;i<=m;i++){int s=i;while(a[i+1].v==a[i].v) i++;for(int j=s;j<=i;j++){x=find(a[j].f),y=find(a[j].to);if(x==y) continue;     g[x].push_back({y,j});g[y].push_back({x,j});}for(int j=s;j<=i;j++){x=find(a[j].f),y=find(a[j].to);if(x==y||dfn[x]) continue;tarjan(x,0);}for(int j=s;j<=i;j++){x=find(a[j].f),y=find(a[j].to);dfn[x]=dfn[y]=0;g[x].clear(),g[y].clear();unite(x,y);}num=0;}int out=0;for(int i=1;i<=m;i++) if(ans[i]) out++;printf("%d\n",out);return 0;
}

 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/440604.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

sharding分表后主键_Sharding-JDBC 使用入门和基本配置

一、什么是Sharding-JDBCSharding-JDBC定位为轻量级Java框架&#xff0c;在Java的JDBC层提供的额外服务。它使用客户端直连数据库&#xff0c;以jar包形式提供服务&#xff0c;无需额外部署和依赖&#xff0c;可理解为增强版的JDBC驱动&#xff0c;完全兼容JDBC和各种ORM框架。…

【蓝桥杯官网试题 - 基础练习】 矩形面积交 (几何)

题干&#xff1a; 问题描述 平面上有两个矩形&#xff0c;它们的边平行于直角坐标系的X轴或Y轴。对于每个矩形&#xff0c;我们给出它的一对相对顶点的坐标&#xff0c;请你编程算出两个矩形的交的面积。 输入格式 输入仅包含两行&#xff0c;每行描述一个矩形。   在每行中…

实现运动轨迹_【自动驾驶】运动规划丨速度规划丨时间维度

运动规划是自动驾驶技术栈中的关键一步&#xff0c;负责把上游的孤立、异构、模糊的多方面信息整合成自洽的运动轨迹&#xff08;trajectory&#xff09;&#xff1b;好的轨迹需要满足多方面的要求&#xff0c;其中最重要的方面包括安全性&#xff08;safety&#xff09;和舒适…

【HDU - 1247】Hat’s Words(字典树,预处理,tricks)

题干&#xff1a; A hat’s word is a word in the dictionary that is the concatenation of exactly two other words in the dictionary. You are to find all the hat’s words in a dictionary. Input Standard input consists of a number of lowercase words, one …

mysql操作json优点和缺点_MySQL与JSON-为什么?

我正在设计一个小的网络应用程序/游戏。 哪个更好&#xff1a;MySQL表或json文件&#xff1f; 它们都存储信息。 它们都可以由PHP解析。 优点/缺点是什么&#xff1f;这就是我的意思&#xff1a;username | password-------------------seefour | abc123与{"username"…

【csust】最小素因子问题(树状数组)

题干&#xff1a; http://csustacm.com:4803/problem/1083 Description 寒冰射手艾希新学会了一个技能&#xff0c;艾希通过这个技能成为了一名声名远扬的神箭手&#xff0c;从此再也无人敢侵犯弗雷尔卓德&#xff01; 这个技能的描述如下&#xff08;假设英雄联盟内的每个…

操作表格_Excel表格基础操作-新手入门级

今日新学了几个基础操作&#xff0c;觉得很有意思&#xff0c;分享给大家。不过真心觉得&#xff0c;有些操作当时学会了&#xff0c;长时间不用&#xff0c;转眼就忘了&#xff1b;另一方面&#xff0c;如果购买相关课程&#xff0c;学着也没什么意思&#xff0c;我前段时间买…

【csust】寻宝(贪心,思维)

题干&#xff1a; Description 在一维坐标轴上有许多宝藏&#xff0c;总共n种宝藏&#xff0c;每种宝藏有k个。现在共k个人寻宝&#xff0c;k个人初始位置可以位于任意点。但是每人需要按指定顺序捡起宝藏&#xff08;1->2->3->...->n&#xff0c;先捡第1种&…

dmsetup remove_all 这命令干啥的_分一个小知识,服务器上的一个解压与压缩文件的命令....

在服务器上对文件做解压操作,是一件非常常见的操作,如安装软件就有很多是要用的到这个操作的.去官网下载一个压缩包,然后解压,编译,再安装.当然网上是能找到相应的命令操作.但是每次都要去找命令,就算记住了,不知道为什么要这样去使用,那么如果不经常使用那还是会忘记的.所以要…

【POJ - 1947】Rebuilding Roads (树形dp,背包问题,树形背包dp)

题干&#xff1a; The cows have reconstructed Farmer Johns farm, with its N barns (1 < N < 150, number 1..N) after the terrible earthquake last May. The cows didnt have time to rebuild any extra roads, so now there is exactly one way to get from any …

seo vue 动态路由_基于vue.jsvue-router的动态更新TDK(SEO优化)

本文基于工作项目开发&#xff0c;做的整理笔记前几天帮朋友解决这个问题&#xff0c;顺带学习了一下&#xff0c;做个笔记Mark下。前提条件&#xff1a;你已经了解并使用vue&#xff0c;能够搭建应用站点。编码环境&#xff1a;system&#xff1a;OS X EI Capitan 10.13.3npm&…

*多叉树的树形背包常见建模方法

一.多叉树变二叉树。 这个技巧其实也有两种具体的方法&#xff1a;树的孩子兄弟表示法与dfs序法。 1.树的孩子兄弟表示法。 大家在学习树形结构时一定接触了一个多叉树变二叉树的方法&#xff0c;就是把每个点与它的第一个儿子连边&#xff0c;然后将它的儿子依次连接起来。可…

python把英语句子成分字母_求一个可以分析英语句子成分的软件或网站

目前还没有&#xff0c;最好的翻译软件都不能翻译英语语法&#xff0c;只能翻译语法结构简单的句子&#xff0c;更别提专业分析句子成分的软件了&#xff1b;出现专业分析英语句子成分的软件&#xff0c;英语老师应该就失业了。求一个可以分析英语句子成分的软件或网站句子是按…

【51nod - 1076】2条不相交的路径(Tarjan无向图判环)

题干&#xff1a; 给出一个无向图G的顶点V和边E。进行Q次查询&#xff0c;查询从G的某个顶点V[s]到另一个顶点V[t]&#xff0c;是否存在2条不相交的路径。&#xff08;两条路径不经过相同的边&#xff09; &#xff08;注&#xff0c;无向图中不存在重边&#xff0c;也就是说…

python 导入csv文件到oracle_python将文件夹下的所有csv文件存入mysql和oracle数据库

#oracle版首先新建python脚本(d:/python/orclImport.py)import os #引入os包if __name__ __main__:#遍历文件夹下所有的dmp文件&#xff0c;其中filename为所有文件、文件夹的名称。#因为我文件夹下确定都是dmp文件&#xff0c;所以无需进行特殊判断for filename in os.listdi…

createform用法_vue自定义表单生成器form-create使用详解

介绍form-create 是一个可以通过 JSON 生成具有动态渲染、数据收集、验证和提交功能的表单生成器。并且支持生成任何 Vue 组件。结合内置17种常用表单组件和自定义组件&#xff0c;再复杂的表单都可以轻松搞定。文档 | github演示项目: 开源的高品质微信商城功能自定义组件可生…

【蓝桥杯官网试题 - 历届试题】发现环(dfs+并查集,或无向图tarjan判环,无向环,或拓扑排序)

题干&#xff1a; 问题描述 小明的实验室有N台电脑&#xff0c;编号1~N。原本这N台电脑之间有N-1条数据链接相连&#xff0c;恰好构成一个树形网络。在树形网络上&#xff0c;任意两台电脑之间有唯一的路径相连。   不过在最近一次维护网络时&#xff0c;管理员误操作使得某…

倍福 在 vs 里 编程 是怎么做到的_截图里的文字要改,字体怎么做到一模一样?...

大家是不是碰到过这么一个尴尬的情况&#xff1a;一个截图里有一个错别字&#xff0c;或者其中一小段文字要修改&#xff0c;可是苦于找不到源文件&#xff0c;又没办法找出和原来文字一模一样的字体。每次遇到这样的情况你都是怎么处理的呢&#xff1f;火箭君今天给大家介绍一…

【ZOJ - 3591】Nim(博弈问题,思维,STLmap)

题干&#xff1a; Nim is a mathematical game of strategy in which two players take turns removing objects from distinct heaps. The game ends when one of the players is unable to remove object in his/her turn. This player will then lose. On each turn, a pla…

【牛客 - 369B】小A与任务(贪心,优先队列)

题干&#xff1a; 链接&#xff1a;https://ac.nowcoder.com/acm/contest/369/B 来源&#xff1a;牛客网 小A手头有 n 份任务&#xff0c;他可以以任意顺序完成这些任务&#xff0c;只有完成当前的任务后&#xff0c;他才能做下一个任务 第 i 个任务需要花费 xixi 的时间…