随笔-68  评论-10  文章-0  trackbacks-0
/*
    sap+gap优化,采用邻接矩正
    时间复杂度:O(mn^2)
*/


#include
<iostream>
#include
<queue>
using namespace std;

const int msize=205;     //最大顶点数
const int inf=0x7fffffff;

int dis[msize];   //标号
int r[msize][msize];   //残留网络,初始为原图
int num[msize];    //num[i]表示标号为i的顶点数有多少
int pre[msize];
int n,m,src,des;    //n个顶点,m条边,源点src,汇点des

void rev_bfs()   //反向bfs计算标号,汇点des标号为0
{
    
int i,k;
    
for(i=1;i<=n;i++)
    
{
        dis[i]
=n;
        num[i]
=0;
    }


    queue
<int> q;
    q.push(des);
    dis[des]
=0;
    num[
0]=1;
    
while(!q.empty())
    
{
        k
=q.front();
        q.pop();
        
for(i=1;i<=n;i++)
        
{
            
if(dis[i]==n&&r[i][k]>0)
            
{
                dis[i]
=dis[k]+1;
                q.push(i);
                num[dis[i]]
++;
            }

        }

    }

}


int findArc(int i)
{
    
for(int j=1;j<=n;j++)
    
{
        
if(r[i][j]>0&&dis[i]==dis[j]+1)
            
return j;
    }

    
return -1;
}


int reLable(int i)
{
    
int mindis=n;
    
for(int j=1;j<=n;j++)
    
{
        
if(r[i][j]>0
            mindis
=mindis<dis[j]? mindis:dis[j];
    }

    
return mindis;
}


int maxFlow()
{
    rev_bfs();

    
int totalflow=0, i=src, j, k;
    
int d;     //增量

    pre[src]
=-1;
    
while(dis[src]<n)
    
{
        j
=findArc(i);
        
if(j>=0)
        
{
            pre[j]
=i;
            i
=j;
            
if(i==des)
            
{
                d
=inf;
                
for(k=des;k!=src;k=pre[k])
                    d
=d<r[pre[k]][k]? d:r[pre[k]][k];
                
for(k=des;k!=src;k=pre[k])
                
{
                    r[pre[k]][k]
-=d;
                    r[k][pre[k]]
+=d;
                }

                totalflow
+=d;
                i
=src;
            }

        }

        
else
        
{
            
--num[dis[i]];
            
if(0==num[dis[i]]) return totalflow;
            
int x=reLable(i);
            dis[i]
=x+1;
            num[dis[i]]
++;
            
if(i!=src) i=pre[i];
        }

    }

    
return totalflow;
}


int main()
{
    
while(scanf("%d%d",&m,&n)!=EOF)
    
{
        
int i;
        
int u,v,w;
        memset(r,
0,sizeof(r));
        
for(i=0;i<m;i++)
        
{
            scanf(
"%d%d%d",&u,&v,&w);
            r[u][v]
+=w;
        }

        src
=1;
        des
=n;
        printf(
"%d\n",maxFlow());
    }

    
return 0;
}

posted on 2011-05-10 13:12 wuxu 阅读(464) 评论(0)  编辑 收藏 引用 所属分类: 图论

只有注册用户登录后才能发表评论。
网站导航: 博客园   IT新闻   BlogJava   知识库   博问   管理