https://cn.vjudge.net/problem/26727/origin

题目大意:

给一个带权的无向图,保证没有自环和重边. 由于最小生成树不唯一,因此你需要确定每一条边是以下三种情况哪一个 1.一定在所有MST上 2.可能在某个MST上 3.一定不可能在任何MST上 输入

第一行给出n,m表示点数和边数. 数据范围见原题面 之后m行,每行给出ai,bi,wi 表示一个边的两个端点和边的权值.保证没有自环与重边. 输出

一看到这题就觉得似曾相似,仔细一看是某场比赛里卡了我三个小时的类似题,当时由于题解写的和我写的算法不一样,没有去补题。这次遇到这题偶然发现由于这题的数据开大了,之前那题的暴力算法走不通了,正好用到了我之前写了很久没找出bug的算法。

第一步显然是将所有边按权值从小到大排序,将每层权值相同的边分开来做。

第二步显然是将每层所有两点已经联通的点打上none的标记

问题在于分开必通过和可以不通过的边,上次只想到了dfs,也想到将所有已联合的点形成联通快找联通快之间必连的点,这次看了题解,发现这种联通块之间必须链接的点被称为桥,必须连接的点成为割点,寻找桥和割点都是通过tarjan算法实现的,也就是基于dfs打一个时间戳实现。

所以我们只要每一层进行见图找出所有的桥即可解决问题。

#include <map>
#include <set>
#include <ctime>
#include <cmath>
#include <queue>
#include <stack>
#include <vector>
#include <string>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <sstream>
#include <iostream>
#include <algorithm>
#include <functional>
using namespace std;
#define For(i, x, y) for(int i=x;i<=y;i++)
#define _For(i, x, y) for(int i=x;i>=y;i--)
#define Mem(f, x) memset(f,x,sizeof(f))
#define Sca(x) scanf("%d", &x)
#define Sca2(x,y) scanf("%d%d",&x,&y)
#define Scl(x) scanf("%lld",&x);
#define Pri(x) printf("%d\n", x)
#define Prl(x) printf("%lld\n",x);
#define CLR(u) for(int i=0;i<=N;i++)u[i].clear();
#define LL long long
#define ULL unsigned long long
#define mp make_pair
#define PII pair<int,int>
#define PIL pair<int,long long>
#define PLL pair<long long,long long>
#define pb push_back
#define fi first
#define se second
typedef vector<int> VI;
const double eps = 1e-;
const int maxn = 1e5 + ;
const int INF = 0x3f3f3f3f;
const int mod = 1e9 + ;
int N,M,tmp,K,tot,index;
int tree[maxn];
int type[maxn];
int vis[maxn];
int head[maxn];
struct Query{
int u,v,w,id;
}q[maxn];
struct Edge{
int v,next,id;
}edge[maxn * ];
bool cmp(Query a,Query b){
return a.w < b.w;
}
int find(int p){
if(p == tree[p]) return p;
return tree[p] = find(tree[p]);
}
void Union(int a,int b){
a = find(a); b = find(b);
tree[a] = b;
}
void add(int u,int v,int id){
edge[tot].v = v;
edge[tot].next = head[u];
edge[tot].id = id;
head[u] = tot++;
}
int dfs(int u,int last){
int lowu = vis[u] = ++index;
for(int i = head[u]; ~i;i = edge[i].next){
int v = edge[i].v;
if(!vis[v]){
int lowv = dfs(v,i);
lowu = min(lowu,lowv);
if(lowv > vis[u]) type[edge[i].id] = ;
}else if(vis[v] < vis[u] && i != (last ^ )){
lowu = min(lowu,vis[v]);
}
}
return lowu;
}
int main()
{
scanf("%d%d",&N,&M);
For(i,,M) scanf("%d%d%d",&q[i].u,&q[i].v,&q[i].w),q[i].id = i;
sort(q + ,q + + M,cmp);
For(i,,N) tree[i] = i;
For(i,,M){
int j = i;
while(j <= M && q[j].w == q[i].w)j++; j--;
For(k,i,j){
q[k].u = find(q[k].u);q[k].v = find(q[k].v);
int u = q[k].u; int v = q[k].v;
if(u == v){
type[q[k].id] = ;
continue;
}
type[q[k].id] = ;
vis[u] = vis[v] = ;
head[u] = head[v] = -;
}
index = tot = ;
For(k,i,j){
if(q[k].u != q[k].v){
add(q[k].u,q[k].v,q[k].id);
add(q[k].v,q[k].u,q[k].id);
}
}
For(k,i,j) if(!vis[q[k].u]) dfs(q[k].u,-);
For(k,i,j) Union(q[k].u,q[k].v);
i = j;
}
For(i,,M){
if(!type[i]) puts("none");
else if(type[i] == )puts("any");
else puts("at least one");
}
#ifdef VSCode
system("pause");
#endif
return ;
}

CodeForces160D 最小生成树 + dfs的更多相关文章

  1. HDU 5723 Abandoned country (最小生成树 + dfs)

    Abandoned country 题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=5723 Description An abandoned coun ...

  2. HDU 5723 Abandoned country (最小生成树+dfs)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=5723 n个村庄m条双向路,从中要选一些路重建使得村庄直接或间接相连且花费最少,这个问题就是很明显的求最 ...

  3. hdu4126(最小生成树+dfs)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4126 题意:给出一幅3000个点的图,有10000次操作: 求将某条边的权值变大后的最小生成树,最后输 ...

  4. HDU 2489 Minimal Ratio Tree 最小生成树+DFS

    Minimal Ratio Tree Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Other ...

  5. 【BZOJ3551】[ONTAK2010]Peaks加强版 最小生成树+DFS序+主席树

    [BZOJ3545][ONTAK2010]Peaks Description 在Bytemountains有N座山峰,每座山峰有他的高度h_i.有些山峰之间有双向道路相连,共M条路径,每条路径有一个困 ...

  6. BZOJ1016:[JSOI2008]最小生成树计数(最小生成树,DFS)

    Description 现在给出了一个简单无向加权图.你不满足于求出这个图的最小生成树,而希望知道这个图中有多少个不同的最小生成树.(如果两颗最小生成树中至少有一条边不同,则这两个最小生成树就是不同的 ...

  7. $bzoj1016-JSOI2008$ 最小生成树计数 最小生成树 $dfs/matrix-tree$定理

    题面描述 现在给出了一个简单无向加权图.你不满足于求出这个图的最小生成树,而希望知道这个图中有多少个不同的最小生成树.(如果两颗最小生成树中至少有一条边不同,则这两个最小生成树就是不同的).由于不同的 ...

  8. HDU 4081 Peach Blossom Spring (最小生成树+dfs)

    题意:给定一个 n 个点和相应的权值,要求你用 n-1 条边连接起来,其中一条边是魔法边,不用任何费用,其他的边是长度,求该魔法边的两端的权值与其他边费用的尽量大. 析:先求出最小生成树,然后再枚举每 ...

  9. cf Inverse the Problem (最小生成树+DFS)

    题意: N个点.N行N列d[i][j]. d[i][j]:结点i到结点j的距离. 问这N个点是否可能是一棵树.是输出YES,否则输出NO. 思路: 假设这个完全图是由一棵树得来的,则我们对这个完全图求 ...

随机推荐

  1. dom定位的三种元素

    1.通过id #XXX 2.通过标签  xxx 3.通过类  .xxx

  2. 牛客网-2018年全国多校算法寒假训练营练习比赛(第四场)-A

    解题思路:二分图的最大匹配,但这题是所有点都遍历一遍,所以答案/2: 代码: #include<iostream> #include<algorithm> #include&l ...

  3. eclipse添加tomcat服务器

    在网上找资料好辛苦,还不对,自己试了好久,终于成功了 还是一如既往的分享 右键 弄好以后发现如此简单| _ |

  4. 前端 -- HTML内容

    HTML介绍 Wed服务本质 import socket sk = socket.socket() sk.bind(("127.0.0.1", 8080)) sk.listen(5 ...

  5. Codeforces519 E. A and B and Lecture Rooms

    传送门:>Here< 题意:询问给出一棵无根树上任意两点$a,b$,求关于所有点$i$,$dist(a,i) = dist(b,i)$的点的数量.要求每一次询问在$O(log n)$的时间 ...

  6. day5 range 用法示例

    函数语法 range(start, stop[, step]) 参数说明: start: 计数从 start 开始.默认是从 0 开始.例如range(5)等价于range(0, 5); stop: ...

  7. mysql 0x80004005 unable to connect to any of the specified mysql hosts

    语言:c# 问题:偶尔会出现连不上mysql 报标题的这个错误. 解决方法:把server = localhost 改为 =127.0.0.1 或者静态IP  ,按着改暂时没出现了,继续观望!

  8. BZOJ4695 最假女选手(势能线段树)

    BZOJ题目传送门 终于体会到初步掌握势能分析思想的重要性了. 一开始看题,感觉套路还是很一般啊qwq.直接在线段树上维护最大值和最小值,每次递归更新的时候,如果不能完全覆盖就暴力递归下去.挺好写的欸 ...

  9. 【BZOJ5335】[TJOI2018]智力竞赛(二分图匹配)

    [BZOJ5335][TJOI2018]智力竞赛(二分图匹配) 题面 BZOJ 洛谷 题解 假装图不是一个DAG想了半天,.发现并不会做. 于是假装图是一个DAG. 那么显然就是二分答案,然后求一个最 ...

  10. SP8791 DYNALCA - Dynamic LCA 解题报告

    SP8791 DYNALCA - Dynamic LCA 有一个森林最初由 \(n (1 \le n \le 100000)\) 个互不相连的点构成 你需要处理以下操作: link A B:添加从顶点 ...