UVa 10735 (混合图的欧拉回路) Euler Circuit
题意:
给出一个图,有的边是有向边,有的是无向边。试找出一条欧拉回路。
分析:
按照往常的思维,遇到混合图,我们一般会把无向边拆成两条方向相反的有向边。
但是在这里却行不通了,因为拆成两条有向边的话,就表示这个边能“在两个相反方向各经过一次”。
而题意是这个边只能经过一次。
假设图中存在欧拉回路,则所有点的出度out(i) 等于 入度in(i)
不妨这样,先将所有的无向边任意定向,对于out(u) > in(u)的点,可以将已经定向的无向边u->v反向为v->u,这样out(u) - in(u)的值减2
如果把出度看做“货物”,则out(u) > in(u)的点提供货物,out(u) < in(u)的点需要货物,所以我们可以用网络流求最大流的算法,来使“供需平衡”
具体来说就是,给已经定向的无向边两点之间连一条容量为1的边,连接源点 与 所有“提供”出度的点,连接 所有“需要”出度的点 与 汇点。
如果求出来的最大流满载,也就是所有的出度都能被运到需要的地方,则有解。
在最大流中,如果流量为1则代表将该边反向的操作。
所以再建一个新图来求欧拉回路。
本题还有一个坑点就是可能存在平行边,所以求欧拉路径的过程中用 vis[u][v] = 1;的方法是行不通的了。
#include <bits/stdc++.h>
#define REP(i,n) for(int i = 0; i < (n); i++)
#define PB push_back
using namespace std; const int INF = ;
const int maxn = + ; struct Edge
{
int from, to, cap, flow;
Edge(int u, int v, int c, int f):from(u), to(v), cap(c), flow(f) {}
}; struct EdmondsKarp
{
int n, m;
vector<Edge> edges;
vector<int> G[maxn];
int a[maxn], p[maxn]; void Init(int n)
{
REP(i, n) G[i].clear();
edges.clear();
} void AddEdge(int from, int to, int cap)
{
edges.PB(Edge(from, to, cap, ));
edges.PB(Edge(to, from, , ));
m = edges.size();
G[from].PB(m-);
G[to].PB(m-);
} int MaxFlow(int s, int t)
{
int flow = ;
for(;;)
{
queue<int> Q;
Q.push(s);
memset(a, , sizeof(a));
a[s] = INF;
while(!Q.empty())
{
int x = Q.front(); Q.pop();
REP(i, G[x].size())
{
Edge& e = edges[G[x][i]];
if(!a[e.to] && e.cap > e.flow)
{
a[e.to] = min(a[x], e.cap - e.flow);
p[e.to] = G[x][i];
Q.push(e.to);
}
}
if(a[t]) break;
}
if(!a[t]) break;
for(int u = t; u != s; u = edges[p[u]].from)
{
edges[p[u]].flow += a[t];
edges[p[u]^].flow -= a[t];
}
flow += a[t];
}
return flow;
}
}g; int n, m;
int deg[maxn], u[maxn], v[maxn], id[maxn];
bool directed[maxn]; vector<int> G[maxn];//建新图,用来求欧拉回路
vector<int> vis[maxn];
vector<int> path;//欧拉回路 void Euler(int u)
{
REP(i, G[u].size()) if(!vis[u][i])
{
vis[u][i] = ;
Euler(G[u][i]);
path.PB(G[u][i]+);
}
} void print_answer()
{
REP(i, n) { G[i].clear(); vis[i].clear(); }
REP(i, m)
{
bool rev = false;
if(!directed[i] && g.edges[id[i]].flow > ) rev = true;//流量为1对应将该边反向
if(!rev) { G[u[i]].PB(v[i]); vis[u[i]].PB(); }
else { G[v[i]].PB(u[i]); vis[v[i]].PB(); }
} path.clear();
Euler();
printf("");
for(int i = path.size()-; i >= ; i--) printf(" %d", path[i]);
puts("");
} int main()
{
//freopen("in.txt", "r", stdin); int T;
scanf("%d", &T);
while(T--)
{
scanf("%d%d", &n, &m);
g.Init(n+);
memset(deg, , sizeof(deg));
REP(i, m)
{
char d;
scanf("%d %d %c", &u[i], &v[i], &d);
u[i]--; v[i]--;
directed[i] = (d == 'D');
deg[u[i]]++; deg[v[i]]--;
if(!directed[i]) { id[i] = g.edges.size(); g.AddEdge(u[i], v[i], ); }//第i条边在网络流中的编号
} bool ok = true;
REP(i, m) if(deg[i] % != ) { ok = false; break; }//出入度之和不是偶数说明不存在欧拉回路 int s = n, t = n+;
if(ok)
{
int sum = ;
REP(i, n)
{
if(deg[i] > ) { sum += deg[i] / ; g.AddEdge(s, i, deg[i] / ); }
if(deg[i] < ) { g.AddEdge(i, t, -deg[i] / ); }
}
int flow = g.MaxFlow(s, t);
if(flow != sum) ok = false;//最大流不满载
} if(ok) print_answer(); else puts("No euler circuit exist");
if(T) puts("");
} return ;
}
代码君
UVa 10735 (混合图的欧拉回路) Euler Circuit的更多相关文章
- 紫书 例题 11-13 UVa 10735(混合图的欧拉回路)(最大流)
这道题写了两个多小时-- 首先讲一下怎么建模 我们的目的是让所有点的出度等于入度 那么我们可以把点分为两部分, 一部分出度大于入度, 一部分入度大于出度 那么显然, 按照书里的思路,将边方向后,就相当 ...
- bzoj2095: [Poi2010]Bridges(二分+混合图求欧拉回路)
传送门 这篇题解讲的真吼->这里 首先我们可以二分一个答案,然后把所有权值小于这个答案的都加入图中 那么问题就转化为一张混合图(既有有向边又有无向边)中是否存在欧拉回路 首先 无向图存在欧拉回路 ...
- [POJ1637]混合图的欧拉回路判定|网络流
混合图的欧拉回路判定 上一篇正好分别讲了有向图和无向图的欧拉回路判定方法 如果遇上了混合图要怎么做呢? 首先我们思考有向图的判定方法:所有点的出度=入度 我们可以先为无向边任意定一个向,算出此时所有顶 ...
- UVA 10735 Euler Circuit 混合图的欧拉回路(最大流,fluery算法)
题意:给一个图,图中有部分是向边,部分是无向边,要求判断是否存在欧拉回路,若存在,输出路径. 分析:欧拉回路的定义是,从某个点出发,每条边经过一次之后恰好回到出发点. 无向边同样只能走一次,只是不限制 ...
- ACM/ICPC 之 混合图的欧拉回路判定-网络流(POJ1637)
//网络流判定混合图欧拉回路 //通过网络流使得各点的出入度相同则possible,否则impossible //残留网络的权值为可改变方向的次数,即n个双向边则有n次 //Time:157Ms Me ...
- POJ 1637 混合图的欧拉回路判定
题意:一张混合图,判断是否存在欧拉回路. 分析参考: 混合图(既有有向边又有无向边的图)中欧拉环.欧拉路径的判定需要借助网络流! (1)欧拉环的判定:一开始当然是判断原图的基图是否连通,若不连通则一定 ...
- bzoj 2095: [Poi2010]Bridges(二分法+混合图的欧拉回路)
[题意] 给定n点m边的无向图,对于边u,v,从u到v边权为c,从v到u的边权为d,问能够经过每条边一次且仅一次,且最大权值最小的欧拉回路. [思路] 二分答案mid,然后切断权值大于mid的边,原图 ...
- POJ1637:Sightseeing tour(混合图的欧拉回路)
Sightseeing tour Time Limit: 1000MS Memory Limit: 10000K Total Submissions: 10581 Accepted: 4466 ...
- POJ 1637 混合图求欧拉回路 最大流实现
前面讲过了无向图,有向图求欧拉回路,欧拉通路的做法.可以直接根据度数来判断,当然前提是这是一个连通图. 这道题既有无向边,又有有向边,然后求欧拉回路. 采用的方法是最大流. 具体处理方法. 首先,我们 ...
随机推荐
- CSS学习------之简单图片切换
最近一直在重温纯CSS,学习的时候真的才发现,css真的博大精深啊! 所以趁着学习的劲头,谢了个最简单的CSS图片切换! 先整理下思路: 首先我希望图片居中间,两边有个切换按钮,点击按钮的时候,可以实 ...
- ios开发之触摸&手势识别
概要: 4个触摸事件.6个手势识别.响应者链条 1.4个触摸事件 1> 触摸事件主要是针对视图的,包括 - (void)touchesBegan:(NSSet *)touches withEve ...
- Linux下tcp协议socket的recv函数返回时机分析(粘包)
http://www.vckbase.com/index.php/wv/10http://blog.csdn.net/zlzlei/article/details/7689409 文章一: 当前在网络 ...
- Win 7怎样拒绝所有可移动存储设备的所有权限
在Windows 7中,我们可拒绝对任何可移动存储类的权限.下面让我来教大家怎样在组策略中启用“所有可移动存储类:拒绝所有权限”策略,具体操作步骤如下所述: 步骤/方法 在开始搜索框中键入“gpedi ...
- 0到N数其中三个数的全排列
#include<iostream> using namespace std; int main(){ ; int count; count=; ;i<=N;i++) ;j<= ...
- java基础知识回顾之javaIO类--管道流PipedOutputStream和PipedIutputStream
管道流(线程通信流):管道流的主要作用是可以进行两个线程间的通讯,分为管道输出流(PipedOutputStream).管道输入流(PipedInputStream),如果想要进行管道输出,则必须要把 ...
- MYSQL存储过程中常使用的命令记录
MYSQL存储过程中常使用的命令记录 1.触发器trigger 查看:show triggers; 2.存储过程procedure 查看:show procedure status; 查看详细:sho ...
- 空格的URL编码
Q: 为什么我看的教材一会说是“+” 一会说是“%20” A: urlencode(" ") '返回+encodeURI(" ") '返回%20是有区别的
- linux入门教程(二) 图形界面还是命令窗口
对于linux的应用,我想大多数都是用在服务器领域,对于服务器来讲真的没有必要跑一个图形界面.所以我们平时安装linux操作系统时往往是不安装图形界面的.说到这里也许你会有疑问,图形界面还能选择装或者 ...
- jackson基于注解的简单使用
Jackson提供了一系列注解,方便对JSON序列化和反序列化进行控制,下面介绍一些常用的注解. 1.@JsonIgnore 此注解用于属性上,作用是进行JSON操作时忽略该属性. 2.@JsonFo ...