• hihoCoder1369 (最大流EK算法,Dinic算法)Ford-Fulkerson


    #1369 : 网络流一·Ford-Fulkerson算法

    时间限制:10000ms
    单点时限:1000ms
    内存限制:256MB

    描述

    小Hi和小Ho住在P市,P市是一个很大很大的城市,所以也面临着一个大城市都会遇到的问题:交通拥挤。

    小Ho:每到周末回家感觉堵车都是一种煎熬啊。

    小Hi:平时交通也还好,只是一到上下班的高峰期就会比较拥挤。

    小Ho:要是能够限制一下车的数量就好了,不知道有没有办法可以知道交通系统的最大承受车流量,这样就可以限制到一个可以一直很顺畅的数量了。

    小Hi:理论上是有算法的啦。早在1955年,T.E.哈里斯就提出在一个给定的网络上寻求两点间最大运输量的问题。并且由此产生了一个新的图论模型:网络流

    小Ho:那具体是啥?

    小Hi:用数学的语言描述就是给定一个有向图G=(V,E),其中每一条边(u,v)均有一个非负数的容量值,记为c(u,v)≥0。同时在图中有两个特殊的顶点,源点S和汇点T。

    举个例子:

    其中节点1为源点S,节点6为汇点T。

    我们要求从源点S到汇点T的最大可行流量,这个问题也被称为最大流问题

    在这个例子中最大流量为5,分别为:1→2→4→6,流量为1;1→3→4→6,流量为2;1→3→5→6,流量为2。

    小Ho:看上去好像挺有意思的,你让我先想想。

    输入

    第1行:2个正整数N,M。2≤N≤500,1≤M≤20,000。

    第2..M+1行:每行3个整数u,v,c(u,v),表示一条边(u,v)及其容量c(u,v)。1≤u,v≤N,0≤c(u,v)≤100。

    给定的图中默认源点为1,汇点为N。可能有重复的边。

    输出

    第1行:1个整数,表示给定图G的最大流。

    样例输入
    6 7
    1 2 3
    1 3 5
    2 4 1
    3 4 2
    3 5 3
    4 6 4
    5 6 2
    样例输出
    5

    分析:hiho的提示讲得很清楚,EK算法或者Dinic算法求最大流。

    Edmond-Karp算法的代码:(88ms)

    #include<cstdio>
    #include<cstring>
    #include<algorithm>
    #include<queue>
    #define INF 999999999
    using namespace std;
    int N,M;//N个点M条边
    int p[502],map[503][503];
    bool vis[502];//标记数组 
    int EK()
    {
        int ans=0;//ans表示最大流,初始化为0 
        while(true)
        {
            queue<int> q;//寻找增广路 
            memset(p,-1,sizeof(p));
            memset(vis,false,sizeof(vis));
            q.push(1);vis[1]=true;
            while(!q.empty())
            {
                int u=q.front();
                if(u==N) break;
                q.pop();
                for(int i=1;i<=N;i++)
                {
                    if(map[u][i]&&!vis[i])//当前边容量非零,且增广点未标记 
                    {
                        vis[i]=true;
                        p[i]=u;//记录点i的前一个结点v
                        q.push(i); 
                    }
                }
            }
            if(p[N]==-1) break;//没有找到增广路 
            int k=N,MAX=INF;//MAX为增广路中的最大流 
            while(p[k]!=-1)
            {
                MAX=min(MAX,map[p[k]][k]);
                k=p[k];
            }
            ans+=MAX;//累加进最大流 
            
            k=N;//修改路径上的边容量 
            while(p[k]!=-1)
            {
                map[p[k]][k]-=MAX;
                map[k][p[k]]+=MAX;
                k=p[k];
            }
        }
        return ans;
    }
    
    
    int main()
    {
        scanf("%d%d",&N,&M);
        for(int i=0;i<M;i++)
        {
            int s,t,c;
            scanf("%d%d%d",&s,&t,&c);
            map[s][t]+=c;
        }
        printf("%d
    ",EK());
        return 0;
    }
    View Code

    Dinic算法的代码:(26ms)

    #include<cstdio>
    #include<cstring>
    #include<algorithm>
    #include<queue>
    #define INF 999999999
    using namespace std;
    int N,M;//N个点M条边 
    int map[502][502];
    int dis[502];//分层 
    
    int bfs()//分层 
    {
        memset(dis,-1,sizeof(dis));
        dis[1]=0;
        queue<int> q;
        q.push(1);
        while(!q.empty())
        {
            int u=q.front();
            q.pop();
            for(int i=1;i<=N;i++)
            if(dis[i]==-1&&map[u][i]>0)
            {
                dis[i]=dis[u]+1;
                q.push(i);
            }
        }
        if(dis[N]>0) return 1;
        return 0;//无法遍历到汇点则返回0退出 
    } 
    
    int dfs(int cur,int m)//m为当前流量上限,即从点cur流出的流量最多为m 
    {
        if(cur==N) return m;
        int f,res=0;
        for(int i=1;i<=N;i++)
        {
            if(dis[i]==dis[cur]+1&&map[cur][i]>0&&(f=dfs(i,min(m,map[cur][i]))))
            {
                map[cur][i]-=f;
                map[i][cur]+=f;//修改路径上的边容量 
                res+=f;
                m-=f;//每找到增广路便修改流量上限 
                if(!m) break;//流量上限为0就退出循环 
            }
        }
        if(res) return res;
        dis[cur]=-1;//res==0的时候这个结点不能流通,即通过该结点找不到增广路 
        return 0;
    }
    
    int main()
    {
        scanf("%d%d",&N,&M);
        for(int i=0;i<M;i++)
        {
            int s,t,c;
            scanf("%d%d%d",&s,&t,&c);
            map[s][t]+=c;
        }
        int ans=0,res;//res为增广路的残余流量 
        while(bfs())
        {
            while(res=dfs(1,INF))
            ans+=res;
        }
        printf("%d
    ",ans);
        return 0;
    }
    View Code

    Dinic算法(数组模拟邻接链表):(17ms)

    #include<cstdio>
    #include<cstring>
    #include<algorithm>
    #define INF 999999999
    #include<queue>
    using namespace std;
    int N,M,cnt;//cnt为边的数量 
    struct Node{
        int s,t,c;
    }edge[50000];//边数是M的最大值的两倍 
    int head[502],next1[50000];
    int dis[502];
    
    void add(int s,int t,int c)
    {
        edge[cnt].s=s;
        edge[cnt].t=t;
        edge[cnt].c=c;
        next1[cnt]=head[edge[cnt].s];
        head[edge[cnt].s]=cnt++;
    }
    
    int bfs()
    {
        memset(dis,-1,sizeof(dis));
        dis[1]=0;
        queue<int> q;
        q.push(1);
        while(!q.empty())
        {
            int u=q.front();
            q.pop();
            int k=head[u];
            while(k!=-1)
            {
                int v=edge[k].t;
                if(dis[v]==-1&&edge[k].c>0)
                {
                    dis[v]=dis[u]+1;
                    q.push(v);
                }
                k=next1[k];
            }
        }
        if(dis[N]>0) return 1;
        return 0;
    }
    
    int dfs(int cur,int m)
    {
        if(cur==N) return m;
        int res=0,f,k=head[cur];
        while(k!=-1)
        {
            if(dis[edge[k].t]==dis[cur]+1&&edge[k].c>0&&(f=dfs(edge[k].t,min(m,edge[k].c))))
            {
                edge[k].c-=f;
                edge[k^1].c+=f;
                res+=f;
                m-=f;
                if(!m) break;
            }
            k=next1[k];
        }
        if(res) return res;
        dis[cur]=-1;
        return 0;
    }
    
    int main()
    {
        scanf("%d%d",&N,&M);
        memset(head,-1,sizeof(head));
        cnt=0;
        for(int i=0;i<M;i++)
        {
            int s,t,c;
            scanf("%d%d%d",&s,&t,&c);
            add(s,t,c);
            add(t,s,0);
        }
        int ans=0,res;
        while(bfs())
            while(res=dfs(1,INF)) ans+=res;
        printf("%d
    ",ans);
        return 0;
    }
    View Code
  • 相关阅读:
    【转载】mysql dual 虚拟表
    Mysql中DATE_SUB函数
    mybatis中<include>标签的作用
    IDEA提升幸福感的小技巧之"快速从controller跳转到serviceImpl"
    MySQL一些常用的时间函数 current_timestamp()
    SpringCloud配置中心搭建与使用(本地存储配置)
    Linux内存管理 (7)VMA操作【转】
    linux下的调试工具ltrace与strace【转】
    (十四)Linux内存管理之page fault处理【转】
    Linux系统调用详解(实现机制分析)--linux内核剖析(六)【转】
  • 原文地址:https://www.cnblogs.com/ACRykl/p/8783211.html
Copyright © 2020-2023  润新知