最大流EK

     #include <iostream>
     #include <queue>      
    #include <algorithm>
    #include<cstring>
     using namespace std;
     const int msize = 205;
       
     int N, M;   // N--路径数, M--结点数
     int r[msize][msize];  //
     int pre[msize];  // 记录结点i的前向结点为pre[i]
     bool vis[msize]; // 记录结点i是否已访问
       
     // 用BFS来判断从结点s到t的路径上是否还有delta
     // 即判断s,t之间是否还有增广路径,若有,返回1
     bool BFS(int s, int t)
     {
         queue<int> que;
         memset(pre, -1, sizeof(pre));
         memset(vis, false, sizeof(vis));
       
         pre[s] = s;
         vis[s] = true;
        que.push(s); //(在栈中加入原点)
       
         int p;
         while(!que.empty()) //()
         {
             p = que.front(); //(栈的开头)
             que.pop(); //(弹出)
             for(int i=1; i<=M; ++i) //(一共m个点)
             {
                 if(r[p][i]>0 && !vis[i]) ()
                 {
                     pre[i] = p;
                     vis[i] = true; //(是否走过)
                     if(i == t)  // 存在增广路径
                         return true;
                     que.push(i);
                 }
             }
         }
         return false;
     }
       
     int EK(int s, int t)
     {
         int maxflow = 0, d;
         while(BFS(s, t)) //(是否有增广路径)
         {
             d= 99999999;
             // 若有增广路径,则找出最小的delta
            for(int i=t; i!=s; i=pre[i])
                 d = min(d, r[pre[i]][i]);
             // 这里是反向边,看讲解
             for(int i=t; i!=s; i=pre[i])
             {
                 r[pre[i]][i] -= d;
                 r[i][pre[i]] += d;
             }
             maxflow += d;
         }
         return maxflow;
     }
       
     int main()
     {
         while(cin >> N >> M)
         {
             memset(r, 0, sizeof(r));
             int s, e, c;
             for(int i=0; i<N; ++i)
            {
                cin >> s >> e >> c;
                 r[s][e] += c;   // 有重边时则加上c
             }
                   cout << EK(1, M) << endl;
        }
        return 0;

   }

将EK算法分为几步,分别是:1:加边,2,BFS找一条增广路,3统计这条路上的“最大流”,答案加上他,加反向边(正的-,反的+),4,返回第2步,直到找不到增广路。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值