• uva10735 Euler Circuit


    题外话:很多混合图问题可以转化为有向图问题(将无向边拆为两条有向边)

    本题不行,因为只能经过一次

    这种问题能想到网络流。。

    复习欧拉回路:入度==出度

    uva1380有点相似,要先给无向边定向。原图为G,定向的边单独组成另一个G

    定向后对任意点,入度==出度,则有了回路。

    否则调整原来的无向边。  (如果入度出度奇偶性不同,则无解)

    出度增加(in-out/2)。

    注意U->V变成V->UU出度-1V出度+1. 就像在运送”出度”,就是网络流。(满足out>in的点能提供出度) (可以提供的出度为diff[i]/2,需要接受的出度为-diff[i]/2

    原有的边cap1(只能改一次方向)

    设立超级节点STS连可以提供的节点,cap为可以提供的出度

    T类似。

    sum可以提供的出度!=MaxFlow,就无解。否则有解(必须提供出去。这里可以证明提供出去后所有的点in==out

    char dir[9];

          scanf("%d%d%s", &u[i], &v[i], dir);    0 1 D

    感觉可以做一个处理输入的专题了...

    #include<cstdio>
    #include<cstring>
    #include<queue>
    #include<algorithm>
    using namespace std;
    
    const int INF = 1000000000;
    
    struct Edge {
      int from, to, cap, flow;
      Edge(int u, int v, int c, int f):from(u),to(v),cap(c),flow(f) {}
    };
    
    const int maxn = 100+10;
    
    struct EdmondsKarp {
      int n, m;
      vector<Edge> edges;    // 边数的两倍
      vector<int> G[maxn];   // 邻接表,G[i][j]表示结点i的第j条边在e数组中的序号
      int a[maxn];           // 当起点到i的可改进量
      int p[maxn];           // 最短路树上p的入弧编号
    
      void init(int n) {
        for(int i = 0; i < n; i++) G[i].clear();
        edges.clear();
      }
    
      void AddEdge(int from, int to, int cap) {
        edges.push_back(Edge(from, to, cap, 0));
        edges.push_back(Edge(to, from, 0, 0));
        m = edges.size();
        G[from].push_back(m-2);
        G[to].push_back(m-1);
      }
    
      int Maxflow(int s, int t) {
        int flow = 0;
        for(;;) {
          memset(a, 0, sizeof(a));
          queue<int> Q;
          Q.push(s);
          a[s] = INF;
          while(!Q.empty()) {
            int x = Q.front(); Q.pop();
            for(int i = 0; i < G[x].size(); i++) {
              Edge& e = edges[G[x][i]];
              if(!a[e.to] && e.cap > e.flow) {
                p[e.to] = G[x][i];
                a[e.to] = min(a[x], e.cap-e.flow);
                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]^1].flow -= a[t];
          }
          flow += a[t];
        }
        return flow;
      }
    };
    
    EdmondsKarp g;
    
    const int maxm = 500 + 5;
    
    int n, m, u[maxm], v[maxm], directed[maxm], id[maxm], diff[maxn];
    
    
    vector<int> G[maxn];
    vector<int> vis[maxn];
    vector<int> path;
    
    void euler(int u) {
      for(int i = 0; i < G[u].size(); i++)
        if(!vis[u][i]) {
          vis[u][i] = 1;
          euler(G[u][i]);
          path.push_back(G[u][i]+1);
        }
    }
    
    void print_answer() {
      // build the new graph
      for(int i = 0; i < n; i++) { G[i].clear(); vis[i].clear(); }
      for(int i = 0; i < m; i++) {
        bool rev = false;
        if(!directed[i] && g.edges[id[i]].flow > 0)   //id记录了无向边在edges中的位置
            rev = true;                                  //G记录边的起点对应的终点
        if(!rev)                          //没有变反向
            { G[u[i]].push_back(v[i]);
              vis[u[i]].push_back(0);
            }
        else {
             G[v[i]].push_back(u[i]);
             vis[v[i]].push_back(0);
             }
      }
    
      // print euler tour
      path.clear();
      euler(0);             //因为节点一记为了0
    
      printf("1");
      for(int i = path.size()-1; i >= 0; i--) printf(" %d", path[i]);
      printf("
    ");
    }
    
    int main() {
      int T;
      scanf("%d", &T);
    
      while(T--) {
        scanf("%d%d", &n, &m);
        g.init(n+2);
    
        memset(diff, 0, sizeof(diff));
        for(int i = 0; i < m; i++) {
          char dir[9];
          scanf("%d%d%s", &u[i], &v[i], dir);
          u[i]--;
          v[i]--;
          directed[i] = (dir[0] == 'D' ? 1 : 0);
          diff[u[i]]++;                             //出度-入度
          diff[v[i]]--;
          if(!directed[i]) { id[i] = g.edges.size(); g.AddEdge(u[i], v[i], 1); }
        }
    
        bool ok = true;
        for(int i = 0; i < n; i++)
          if(diff[i] % 2 != 0) { ok = false; break; }
    
        int s = n, t = n+1;
        if(ok) {
          int sum = 0;
          for(int i = 0; i < n; i++) {
    
            if(diff[i] > 0) {
            g.AddEdge(s, i, diff[i]/2);
            sum += diff[i]/2;
             }
    
            if(diff[i] < 0) {
                    g.AddEdge(i, t, -diff[i]/2); }
             }
    
          if(g.Maxflow(s, t) != sum)
            ok = false;
        }
    
        if(!ok)
            printf("No euler circuit exist
    ");
        else
            print_answer(); // underlying graph is always connected
    
        if(T)
            printf("
    ");
      }
      return 0;
    }
  • 相关阅读:
    文学、哲学段子
    文学、哲学段子
    js技术要点---JS 获取网页源代码
    泛型类,泛型方法,泛型委托的定义方法
    数组元素的逆序数
    stm32 ARM中的RO、RW和ZI DATA
    poj 3040 Allowance 贪心
    schedule()函数的调用时机(周期性调度)
    以JTextPanel为例Swing的鼠标事件详解
    实习生面试总结
  • 原文地址:https://www.cnblogs.com/lqerio/p/9860934.html
Copyright © 2020-2023  润新知