首页 > 其他 > 详细

●小集训之旅 三

时间:2017-03-31 21:33:36      阅读:255      评论:0      收藏:0      [点我收藏+]

2017.3.31

学习内容:网络流之求解最大流算法

引:最大流问题(maximum flow problem),一种组合最优化问题,就是要讨论如何充分利用装置的能力,使得运输的流量最大,以取得最好的效果。

●算法(e,我学了的……)

  • 枚举算法:
    • 由最小切割=最大流定理而来的暴力求法:
    • 枚举所有割来求最小割:
    • 代码图:
    • 技术分享
    • (资料上的东西,提供一种思路,正确性未知。)
  • 增广路算法:
  • 深度优先搜索(最大流 Ford-Fulkerson方法DFS实现):
    • 不断从源点开始深搜增广路,直到没有了增广路为止。
    • 特点:一旦搜索到一条增广路(能到达汇点且该路径的流量大于一)就回溯。
    • 代码:
    • int dfs(int x,int low)
      {
          if(x==m) return low;
          if(vis[x]) return 0;
          vis[x]=1;
          for(int flow,i=1;i<=m;i++)
          { 
              if(tub[x][i]&&(flow=dfs(i,min(low,tub[x][i]))))
              {
                  tub[x][i]-=flow;
                  tub[i][x]+=flow;
                  return flow;
              }
          }
          return 0;
      }
      int maxflow(int s,int t)
      {
          int ans=0,flow;
          while(flow=dfs(1,INF)){memset(vis,0,sizeof(vis));;ans+=flow;}
          return ans;
      }
    • 注意到每一大次dfs中(即从maxflow()调用),对于点x来说,i都是从1开始枚举,浪费了不少时间。我们便可以用一个cur[ ]数组来优化一下:
      int dfs(int x,int low)
      {
          if(x==m) return low;
          if(vis[x]) return 0;
          vis[x]=1;
          for(int flow,i=cur[x];i<=m;i++)//枚举时从cur[x]或cur[x]+1开始似乎都可以,因为疯狂的while()会直到没有增广路才停下来;
          { 
              if(tub[x][i]&&(flow=dfs(i,min(low,tub[x][i]))))
              {
                  cur[x]=i;
                  tub[x][i]-=flow;
                  tub[i][x]+=flow;
                  return flow;
              }
          }
          return 0;
      }
      int maxflow(int s,int t)
      {
          int ans=0,flow;
          while(flow=dfs(1,INF))
          {memset(vis,0,sizeof(vis));memset(cur,0,sizeof(cur));ans+=flow;}
          return ans;
      }
    • 图示cur[ ]作用:
    • 技术分享
  • Dinic算法
  • 利用层次图进行的求最大流算法。
  • 先进行bfs搜索建立层次图(计算每个点的深度),再dfs找增广路(每个点只能由深度比其小1的点搜索而来;把每个点在当层次图下的所有的增广路找完)
  • 也可用cur[ ]优化
  • 代码(Dinic+邻接链表)
  • bool bfs()
    {
        memset(vis,0,sizeof(vis));
        queue<int>q; q.push(1); d[1]=1; vis[1]=1;
        while(!q.empty())
        {
            int u=q.front();q.pop();
            for(int i=head[u];~i;i=e[i].next)if(e[i].capo) 
            {
                int v=e[i].to; if(vis[v]) continue;
                d[v]=d[u]+1; vis[v]=1; q.push(v);
            }
        }
        return vis[m];
    }
    int dfs(int u,int res)
    {
        if(u==m||!res) return res; int flowout=0,v,f;
        for(int &i=cur[u];~i;i=e[i].next)
        {
            v=e[i].to; if(d[v]!=d[u]+1) continue;
            f=dfs(v,min(e[i].capo,res));
            e[i].capo-=f; e[i^1].capo+=f; 
            flowout+=f; res-=f; if(!res) break;
        }
        return flowout;
    }
    int maxflow()
    {
        while(bfs())
        {
            for(int i=1;i<=m;i++) cur[i]=head[i];
            ans+=dfs(1,INF);
        }
    }
  • 注意上面两种算法的dfs的不同(前者是找到一条就猥琐的return了,后者是把该层次图下该点所有的增广路找完后,再return)

●小集训之旅 三

原文:http://www.cnblogs.com/zj75211/p/6653537.html

(0)
(0)
   
举报
评论 一句话评论(0
关于我们 - 联系我们 - 留言反馈 - 联系我们:wmxa8@hotmail.com
© 2014 bubuko.com 版权所有
打开技术之扣,分享程序人生!