poj1637--Sightseeing tour(最大流)

最大流求混合图是否存在欧拉回路。

以下内容摘自http://www.cnblogs.com/Missa/archive/2012/12/05/2803107.html 讲的很清楚。

混合图的欧拉回路问题

 欧拉回路问题。

1 定义 欧拉通路 (Euler tour)——通过图中每条边一次且仅一次,并且过每一顶点的通路。 欧拉回路 (Euler  circuit)——通过图中每条边一次且仅一次,并且过每一顶点的回路。 欧拉图——存在欧拉回路的图。

2 无向图是否具有欧拉通路或回路的判定 G有欧拉通路的充分必要条件为:G 连通,G中只有两个奇度顶点(它们分别是欧拉通路的两个端点)。 G有欧拉回路(G为欧拉图):G连通,G中均为偶度顶点。

3 有向图是否具有欧拉通路或回路的判定 D有欧拉通路:D连通,除两个顶点外,其余顶点的入度均等于出度,这两个特殊的顶点中,一个顶点的入度比出度大1,另一个顶点的入度比出度小1。 D有欧拉回路(D为欧拉图):D连通,D中所有顶点的入度等于出度。

4 混合图。混合图也就是无向图与有向图的混合,即图中的边既有有向边也有无向边。

5 混合图欧拉回路 混合图欧拉回路用的是网络流。

  把该图的无向边随便定向,计算每个点的入度和出度。如果有某个点出入度之差为奇数,那么肯定不存在欧拉回路。因为欧拉回路要求每点入度 = 出度,也就是总度数为偶数,存在奇数度点必不能有欧拉回路。 现在每个点入度和出度之差均为偶数。将这个偶数除以 2,得 x。即是说,对于每一个点,只要将 x 条边反向(入>出就是变入,出>入就是变出),就能保证出 = 入。如果每个点都是出 = 入,那么很明显,该图就存在欧拉回路。 现在的问题就变成了:该改变哪些边,可以让每个点出 = 入?构造网络流模型。有向边不能改变方向,直接删掉。开始已定向的无向边,定的是什么向,就把网络构建成什么样,边长容量上限 1。另新建 s 和 t。对于入 > 出的点 u,连接边(u, t)、容量为 x,对于出 > 入的点 v,连接边(s, v),容量为 x(注意对不同的点x不同)。之后,察看是否有满流的分配。有就是能有欧拉回路,没有就是没有。查看流值分配,将所有流量非 0(上限是 1,流值不是 0 就是1)的边反向,就能得到每点入度 = 出度的欧拉图。 由于是满流,所以每个入 > 出的点,都有 x 条边进来,将这些进来的边反向,OK,入 = 出了。对于出 > 入的点亦然。那么,没和 s、t 连接的点怎么办?和s 连接的条件是出 > 入,和 t 连接的条件是入 > 出,那么这个既没和 s 也没和 t 连接的点,自然早在开始就已经满足入 = 出了。那么在网络流过程中,这些点属于“中间点”。我们知道中间点流量不允许有累积的,这样,进去多少就出来多少,反向之后,自然仍保持平衡。 所以,就这样,混合图欧拉回路问题,解了。

代码:

/***********************************************
Problem: 1637		User: G_lory
Memory: 328K		Time: 141MS
Language: C++		Result: Accepted
***********************************************/
#include <iostream>
#include <cstdio>
#include <queue>
#include <cstring>
#define pk puts("kk");

using namespace std;

const int N = 205;
const int INF = 0x7fffffff;

int cap[N][N];
int flow[N];
int pre[N];
queue<int> q;

int bfs(int src, int des)
{
    while (!q.empty()) q.pop();
    memset(pre, -1, sizeof pre);
    q.push(src);
    flow[src] = INF;
    while (!q.empty())
    {
        int idx = q.front();
        if (idx == des) break;
        q.pop();
        for (int i = 0; i <= des; ++i)
        {
            if (pre[i] == -1 && cap[idx][i] > 0)
            {
                flow[i] = min(flow[idx], cap[idx][i]);
                pre[i] = idx;
                q.push(i);
            }
        }
    }
    if (pre[des] == -1) return -1;
    return flow[des];
}

int maxFlow(int src, int des)
{
    int ans = 0;
    int in;
    while ((in = bfs(src, des)) != -1)
    {
        int k = des;
        while (k != src)
        {
            int last = pre[k];
            cap[last][k] -= in;
            cap[k][last] += in;
            k = last;
        }
        ans += in;
    }
    return ans;
}

int in[N];
int main()
{
    int t;
    scanf("%d", &t);
    while (t--)
    {
        int n, m;
        memset(cap, 0, sizeof cap);
        memset(flow, 0, sizeof flow);
        memset(in, 0, sizeof in);
        scanf("%d%d", &n, &m);
        int a, b, c;

        while (m--)
        {
            scanf("%d%d%d", &a, &b, &c);
            in[a]--; in[b]++;
            if (c == 0) cap[a][b]++;
        }

        int flag = 1;
        for (int i = 1; i <= n; ++i)
        {
            if (in[i] & 1)
            {
                flag = 0;
                break;
            }
        }
        if (!flag)
        {
            puts("impossible");
            continue;
        }
        int sum = 0;
        for (int i = 1; i <= n; ++i)
        {
            if (in[i] < 0) cap[0][i] += ((-in[i]) >> 1);
            else if (in[i] > 0)
            {
                sum += (in[i] >> 1);
                cap[i][n + 1] += (in[i] >> 1);
            }
        }
        if (sum == maxFlow(0, n + 1)) puts("possible");
        else puts("impossible");
    }
    return 0;
}

  

时间: 2024-10-26 07:36:34

poj1637--Sightseeing tour(最大流)的相关文章

poj1637 Sightseeing tour[最大流+欧拉回路]

混合图的欧拉回路定向问题. 顺便瞎说几句,有向图定欧拉回路的充要条件是每个点入度等于出度,并且图联通.无向图的话只要联通无奇点即可. 欧拉路径的确定应该是无向图联通且奇点数0个或2个,有向图忘了,好像复杂一点,这个真考到就暴力瞎搜吧. 既然每个点的度数都定了,又入度等于出度,那两者对半分,在二分图里左向右连上原图的边,左点集与s连容量为待补充的出度,右点集反之.这样如果我真可以定下来的话,就会有左边所有连边都满流.所以跑最大流看能不能到满流(就是差的总出度)即可. 1 #include<iost

poj1637 Sightseeing tour,混合图的欧拉回路问题,最大流解

混合图的欧拉回路问题 题目地址 欧拉回路问题 1 定义 欧拉通路 (Euler tour)--通过图中每条边一次且仅一次,并且过每一顶点的通路. 欧拉回路 (Euler  circuit)--通过图中每条边一次且仅一次,并且过每一顶点的回路. 欧拉图--存在欧拉回路的图.  2 无向图是否具有欧拉通路或回路的判定  G有欧拉通路的充分必要条件为:G 连通,G中只有两个奇度顶点(它们分别是欧拉通路的两个端点). G有欧拉回路(G为欧拉图):G连通,G中均为偶度顶点.  3 有向图是否具有欧拉通路或

POJ1637 Sightseeing tour (混合图欧拉回路)(网络流)

Sightseeing tour Time Limit: 1000MS   Memory Limit: 10000K Total Submissions: 9100   Accepted: 3830 Description The city executive board in Lund wants to construct a sightseeing tour by bus in Lund, so that tourists can see every corner of the beauti

POJ1637:Sightseeing tour(混合图的欧拉回路)

Sightseeing tour Time Limit: 1000MS   Memory Limit: 10000K Total Submissions: 10581   Accepted: 4466 题目链接:http://poj.org/problem?id=1637 Description: The city executive board in Lund wants to construct a sightseeing tour by bus in Lund, so that touri

poj1637 Sightseeing tour 混合图欧拉回路判定

传送门 第一次做这种题, 尽管ac了但是完全不知道为什么这么做. 题目就是给一些边, 有向边与无向边混合, 问你是否存在欧拉回路. 做法是先对每个点求入度和出度, 如果一条边是无向边, 就随便指定一个方向, 然后连一条边, 权值为1. 最后统计入度出度, 如果一个点的(入度-出度)%2==1, 就说明不存在欧拉回路. 如果全都满足, 就判断每个点的入度出度的大小关系, 入度>出度, 就向汇点连一条边, 权值为(入度-出度)/2, 相反的话就向源点连边. 跑一遍最大流, 看是否满流, 如果满流就说

POJ1637 Sightseeing tour(判定混合图欧拉回路)

有向连通图存在欧拉回路的充要条件是所有点入度=出度. 首先随便给定所有无向边一个方向(不妨直接是u->v方向),记录所有点的度(记:度=入度-出度). 这时如果有点的度不等于0,那么就不存在欧拉回路,就需要改变那些无向边的方向. 而改变一个无向边的方向,相当于边上两个端点的入度和出度都变化了1,它们的度±2. 另外,这样可以证明如果这时某个点的度为奇数那么一定不存在存在欧拉回路的解. 构图如下:所有无向边(u,v),建立容量为1的(u,v)边:所有度小于0的点u,建立容量为-deg/2的(vs,

poj 1637 Sightseeing tour —— 最大流+欧拉回路

题目:http://poj.org/problem?id=1637 建图很妙: 先给无向边随便定向,这样会有一些点的入度不等于出度: 如果入度和出度的差值不是偶数,也就是说这个点的总度数是奇数,那么一定无解: 随便定向后,如果定向 x -> y,那么从 y 向 x 连一条容量为1的边,将来选了这条边,表示重新定向成 y -> x 了: 考虑如果选了这条边,那么 x 的出度-1,入度+1,变化量是2: 所以对于每个点,如果入度>出度,从源点向它连容量为 (入度-出度)/2 的边,因为刚才改

【POJ1637】Sightseeing tour

Description The city executive board in Lund wants to construct a sightseeing tour by bus in Lund, so that tourists can see every corner of the beautiful city. They want to construct the tour so that every street in the city is visited exactly once.

POJ 1637 Sightseeing tour(最大流)

POJ 1637 Sightseeing tour 题目链接 题意:给一些有向边一些无向边,问能否把无向边定向之后确定一个欧拉回路 思路:这题的模型很的巧妙,转一个http://blog.csdn.net/pi9nc/article/details/12223693 先把有向边随意定向了,然后依据每一个点的入度出度之差,能够确定每一个点须要调整的次数,然后中间就是须要调整的边,容量为1,这样去建图最后推断从源点出发的边是否都满流就可以 代码: #include <cstdio> #includ

POJ 1637 Sightseeing tour (混合图欧拉回路,网络最大流)

http://poj.org/problem?id=1637 Sightseeing tour Time Limit: 1000MS   Memory Limit: 10000K Total Submissions: 7498   Accepted: 3123 Description The city executive board in Lund wants to construct a sightseeing tour by bus in Lund, so that tourists can