HDU 1956 POJ 1637 Sightseeing tour
混合图的欧拉回路判定方法:
1.首先判断基图是否连通,不连通的话表示不可能,否则进入下一步。
2.对于无向边,随便确定一个方向
3.确定好了之后,整张图就变成了有向图,计算每个节点的入度与出度
4.如果有一个节点的入度—出度是奇数,那么表示不可能,否则进入下一步
5.建立网络,新增一个原点s,和汇点t,然后建立网络
for(i=; i<=M; i++)
if(ff[i]==)//如果是有向边
AddEdge(u[i],v[i],);
for(i=; i<=N; i++)
{
if(Ru[i]>Chu[i])
AddEdge(i,t,(Ru[i]-Chu[i])/);
else
AddEdge(s,i,(Chu[i]++-Ru[i])/);
}
6.计算网络最大流。
7.如果从S引出的边有流量的都是满流,那么表示存在,否则不存在。
8.把网络流中与S,T不关联的边找到,这些边中如果有流量等于1的边,那么将这些边反向,最终得到了一张欧拉图。
AC代码(网络最大流用了Dinic连续最短增广路算法):
#include<cstdio>
#include<cstring>
#include<string>
#include<cmath>
#include<vector>
#include<queue>
#include<algorithm>
using namespace std; const int maxn=+;
const int INF=0x7FFFFFFF; struct Edge
{
int from,to,cap,flow;
};
vector<Edge>edges;
vector<int>G[maxn];
bool vis[maxn];
int d[maxn];
int cur[maxn];
int Ru[maxn];
int Chu[maxn];
int u[maxn],v[maxn],ff[maxn];
int father[maxn];
int m,s,t,tot;
int N,M; //求出层次网络
bool BFS()
{
memset(vis,,sizeof(vis));
queue<int>Q;
Q.push(s);
d[s]=;
vis[s]=;
while(!Q.empty())
{
int x=Q.front();
Q.pop();
for(int i=; i<G[x].size(); i++)
{
Edge& e=edges[G[x][i]];
if(!vis[e.to]&&e.cap>e.flow)
{
vis[e.to]=;
d[e.to]=d[x]+;
Q.push(e.to);
}
}
}
return vis[t];
} //加边
void AddEdge(int from,int to,int cap)
{
Edge r;
r.from=from;
r.to=to;
r.cap=cap;
r.flow=;
edges.push_back(r);
Edge d;
d.from=to;
d.to=from;
d.cap=;
d.flow=;
edges.push_back(d);
m=edges.size();
G[from].push_back(m-);
G[to].push_back(m-);
} //每个阶段来一次DFS增广
int DFS(int x,int a)
{
if(x==t||a==) return a;
int flow=,f;
for(int i=cur[x]; i<G[x].size(); i++)
{
Edge& e=edges[G[x][i]];
if(d[x]+==d[e.to]&&(f=DFS(e.to,min(a,e.cap-e.flow)))>)
{
e.flow+=f;
edges[G[x][i]^].flow-=f;
flow+=f;
a-=f;
if(a==) break;
}
}
return flow;
} //多个阶段,多次建立层次网络。
int Maxflow(int ss,int tt)
{
int flow=;
while(BFS())
{
memset(cur,,sizeof(cur));
flow+=DFS(ss,INF);
}
return flow;
} int Find(int x)
{
if(x!=father[x]) father[x]=Find(father[x]);
return father[x];
} int main()
{
int T,flag,i;
scanf("%d",&T);
while(T--)
{
scanf("%d%d",&N,&M);
edges.clear();
for(i=; i<maxn; i++) G[i].clear();
flag=;
s=,t=N+;//设置超级原点和超级汇点
memset(Ru,,sizeof(Ru));
memset(Chu,,sizeof(Chu));
for(i=;i<=N;i++) father[i]=i;
tot=N;
for(i=; i<=M; i++)
{
scanf("%d%d%d",&u[i],&v[i],&ff[i]);
int fx=Find(u[i]);
int fy=Find(v[i]);
if(fx!=fy)
{
father[fx]=fy;
tot--;
}
Ru[v[i]]++;
Chu[u[i]]++;
}
if(tot!=) flag=;
if(flag)
{
for(i=; i<=N; i++)
if(abs(Ru[i]-Chu[i])%==)
{
flag=;
break;
}
}
if(flag)
{
for(i=; i<=M; i++)
if(ff[i]==)//如果是有向边
AddEdge(u[i],v[i],);
for(i=; i<=N; i++)
{
if(Ru[i]>Chu[i])
AddEdge(i,t,(Ru[i]-Chu[i])/);
else
AddEdge(s,i,(Chu[i]++-Ru[i])/);
}
Maxflow(s,t);
for(i=; i<edges.size(); i++)
if(edges[i].from==s&&edges[i].cap!=edges[i].flow)
{
flag=;
break;
}
}
if(flag) printf("possible\n");
else printf("impossible\n");
}
return ;
}
HDU 1956 POJ 1637 Sightseeing tour的更多相关文章
- POJ 1637 Sightseeing tour(最大流)
POJ 1637 Sightseeing tour 题目链接 题意:给一些有向边一些无向边,问能否把无向边定向之后确定一个欧拉回路 思路:这题的模型很的巧妙,转一个http://blog.csdn.n ...
- POJ 1637 Sightseeing tour (混合图欧拉路判定)
Sightseeing tour Time Limit: 1000MS Memory Limit: 10000K Total Submissions: 6986 Accepted: 2901 ...
- POJ 1637 - Sightseeing tour - [最大流解决混合图欧拉回路]
嗯,这是我上一篇文章说的那本宝典的第二题,我只想说,真TM是本宝典……做的我又痛苦又激动……(我感觉ACM的日常尽在这张表情中了) 题目链接:http://poj.org/problem?id=163 ...
- POJ 1637 Sightseeing tour
Sightseeing tour Time Limit: 1000MS Memory Limit: 10000K Total Submissions: 9276 Accepted: 3924 ...
- POJ 1637 Sightseeing tour (混合图欧拉回路)
Sightseeing tour Description The city executive board in Lund wants to construct a sightseeing tou ...
- 网络流(最大流) POJ 1637 Sightseeing tour
Sightseeing tour Time Limit: 1000MS Memory Limit: 10000K Total Submissions: 8628 Accepted: 3636 ...
- POJ 1637 Sightseeing tour (SAP | Dinic 混合欧拉图的判断)
Sightseeing tour Time Limit: 1000MS Memory Limit: 10000K Total Submissions: 6448 Accepted: 2654 ...
- POJ 1637 Sightseeing tour(混合图欧拉回路+最大流)
http://poj.org/problem?id=1637 题意:给出n个点和m条边,这些边有些是单向边,有些是双向边,判断是否能构成欧拉回路. 思路: 构成有向图欧拉回路的要求是入度=出度,无向图 ...
- poj 1637 Sightseeing tour——最大流+欧拉回路
题目:http://poj.org/problem?id=1637 先给无向边随便定向,如果一个点的入度大于出度,就从源点向它连 ( 入度 - 出度 / 2 ) 容量的边,意为需要流出去这么多:流出去 ...
随机推荐
- [bzoj2957][楼房重建] (线段树)
Description 小A的楼房外有一大片施工工地,工地上有N栋待建的楼房.每天,这片工地上的房子拆了又建.建了又拆.他经常无聊地看着窗外发呆,数自己能够看到多少栋房子. 为了简化问题,我们考虑这些 ...
- Objective-C运行时态消息传递--拼接方法名
做IOS开发的人都知道,Objective-C语言中方法的调用是运行时采取绑定的,在编译过程中只声明该方法的存在. 那么我们来简单说下在运行时,类的消息传递. 在运行时,每个方法如[self meth ...
- jquery通过attr取html里自定义属性原来这么方便啊
<script type="text/javascript"> function fangGouWuChe(obj) { //放入购物车 var sMat = $(o ...
- C# FTPHelper(搬运)
using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.N ...
- 关于Two-Pass标记连通域个数
关于Two-Pass标记连通域个数 背景 在完成图像的一系列处理后,得到二值图,一般会统计目标数量,即是获取连通域个数,这里采用TwoPass的方法. 基本思想 在Two-pass连通域标记中,第一次 ...
- [转]numpy性能优化
转自:http://blog.csdn.net/pipisorry/article/details/39087583 http://blog.csdn.net/pipisorry/article/de ...
- 如何运用xp_cmdshell 执行多行cmd命令
因为需要利用sql 脚本调用外部程序,所以尝试xp_cmdshell 命令,发现其默认只能执行一行命令,最后找到了下面方法: 将多行cmd 命令处理成bat文件,然后直接利用所在路径+文件来执行: ...
- 项目中dubbo的使用
导语:Dubbo是阿里巴巴的一个分布式服务的开源框架,致力于提供高性能和透明化的RPC远程服务调用方案,是阿里巴巴SOA服务化治理方案的核心框架,每天为2,000+个服务提供3,000,000,000 ...
- 使用canvas实现超绚丽的旋转正方形
自己无意中的一个小"bug",却让动画变得超绚丽= = 所以,不要害怕出bug,谁知道bug不会开出一朵绚丽的花呢? <!DOCTYPE html> <html ...
- Ansible Filter
没啥好说的. 自己看吧 http://docs.ansible.com/ansible/playbooks_filters.html