Dijkstra的最小费用最大流问题

对于最小费用最大流问题,它的重点就在于   “增广路”    什么是  增广路? 就是在以找的的路的基础上再加一条路 加上这条路能让结果更大,直接使用 Dijkstra 能找的的路是最短的路,继续用能找到剩下路中间的最小路, 但是这两条路加上来不一定是总体的最小路 ,第一次 1->3 ->5->4->6 第二次1->2->6 这并不是我们需要的结果,所以在找第二条最短路时,我们需要能反悔,能不

让第一次不走3->5,怎么反悔,在构建图时加上一条反边,第一次走了多少正边减少多少反边加上多少,这样第二次就可以走5->3然后走3->6这条线,第一次走了3->5,第二次凑了5->3相当与反悔了第一次的路,这样就能尽可能的走最小路。

#include<iostream>
#include<cstring>
#include<queue>
using namespace std;
/*6 11 1 2 23 1 3 12 1 4 99 2 5 17 2 6 73 3 5 3 3 6 21 4 6 8 5 2 33 5 4 5 6 5 20*/
#define MAX 23060
int MinCos;
int v,cnt;
int Head[MAX];
int Next[MAX];
int F[MAX];//流量
int To[MAX];//终结的
int Dis[MAX];
int Cos[MAX];//花费
int vis[MAX];
void _add(int a,int b,int f,int c)
{
    cnt++;
    F[cnt]=f;
    To[cnt]=b;
    Cos[cnt]=c;
    Next[cnt]=Head[a];
    Head[a]=cnt;
}
void add(int a,int b,int c)//增加点 用于构图
{
    if(!vis[a]&&a!=1&&a!=v)
    {
        _add(a,a+v,1,0);
        _add(a+v,a,0,0);
        vis[a]=1;
    }
    if(!vis[b]&&b!=1&&b!=v)
    {
        _add(b,b+v,1,0);
        _add(b+v,b,0,0);
        vis[b]=1;
    }
    if(a==1||b==v)
    {
        if(a!=1&&b==v)
        {
            _add(a+v,b,1,c);
            _add(b,a+v,0,-c);
        }else if(a==1&&b!=v)
        {
            _add(a,b,1,c);
            _add(b,a,0,-c);
        }
        else
        {
            _add(a,b,1,c);
            _add(b,a,0,-c);
        }
    }else if(a==v||b==1){}
    else
    {
        _add(a+v,b,1,c);
        _add(b,a+v,0,-c);
    }
}
void Dijks(int folw)//Dijks求最小费用最大流算法
{
    priority_queue<pair<int,int>,vector<pair<int,int> >,greater<pair<int,int> > >sp;
    int h[MAX];
    To[MAX-1]=1;
    memset(h,0,sizeof(h));
    while(folw>0) {
        memset(Dis, 0x3f3f3f3f, sizeof(Dis));
        memset(vis,0,sizeof(vis));
        Dis[1] = 0;
        pair<int, int> t;
        sp.push(make_pair(0,MAX-1));
        int pre[MAX];
        pre[1] = 0;
        while (!sp.empty()) {
            t = sp.top();
            sp.pop();
            int to = To[t.second];
            if(to==v)continue;
            if(vis[to])continue;
            if (Dis[to] < t.first)continue;//剪枝
            vis[to]=1;
            for (int i = Head[to]; i != -1; i = Next[i]) {
                if (F[i] && Dis[To[i]] > Dis[to] + h[to] - h[To[i]] + Cos[i]) {//跟新最小费用
                    Dis[To[i]] = Dis[to] + h[to] - h[To[i]] + Cos[i];
                    pre[i]=t.second;//记录上一个节点
                    if(To[i]==v)
                    {
                        pre[MAX-2]=i;
                    }
                    sp.push(make_pair(Dis[To[i]], i));
                }
            }
        }
        if(Dis[v]==0x3f3f3f3f)break;
        for (int i = 1; i <= 2*v; i++) {
            h[i] += Dis[i];//我也不是能理解但是有篇博客写的特别好
        }
        for(int i=pre[MAX-2];i!=MAX-1;i=pre[i])//求最小费用
        {
            MinCos+=Cos[i];
            F[i] -= 1;//存入数据时0开始,偶数存正向边,奇数反向边
            F[i ^ 1] += 1;//i^1就是奇数
        }
        folw -= 1;
    }
    cout<<MinCos<<"\n";
}
int main()
{
    ios::sync_with_stdio(false);
    cin.tie(0);
    int p;
    while(cin>>v>>p) {
        for(int i=0;i<20060;i++)
        {
            F[i]=0;
            To[i]=0;
            Dis[i]=0;
            Cos[i]=0;
            vis[i]=0;
            Head[i]=-1;
            Next[i]=-1;
        }
        MinCos=0;
        cnt=-1;
        for (int i = 0; i < p; i++) {
            int a, b, c;
            cin >> a >> b >> c;
            add(a, b, c);
        }
        Dijks(2);
    }
    return 0;
}

一个大佬讲解h[i]函数的博客http://www.cppblog.com/guojingjia2006/archive/2009/11/12/57905.html

原文地址:https://www.cnblogs.com/hycn/p/11329955.html

时间: 2024-10-07 03:39:50

Dijkstra的最小费用最大流问题的相关文章

hdu5352 MZL&#39;s City(最小费用最大流问题)

题意描述: 一个国家有N个城市,标号1~N,初始时城市和道路都被破坏,下面进行以下三个操作: 第一种操作:1 X 修建城市,和X(包括X自己也可以被修建)直接相连或间接相连的城市可以被一次性修建(但一次性最多修建k个城市) 第二种操作:2 X Y 在X 与 Y之间建立一条道路 第三种操作:3 p X1 Y1 X2 Y2 ··· 表示破坏X1与Y1.X2与Y2之间的道路 经过M次操作后,对于这M次操作中的操作一,我们在每次操作一时修建多少城市修建那些城市才能使最终城市数量最多? 并输出最终修建城市

poj3422 Kaka&#39;s Matrix Travels(最小费用最大流问题)

1 /* 2 poj3422 Kaka's Matrix Travels 3 不知道 k次 dp做为什么不对??? 4 看了大牛的代码,才知道还可以这样做! 5 开始没有理解将a 和 a‘ 之间建立怎样的两条边,导致程序一直陷入死循环,真心花了好长时间,快崩溃了.无语..... 6 题意:有个方阵,每个格子里都有一个非负数,从左上角走到右下角,每次走一步,只能往右或往下走,经过的数字拿走 7 每次都找可以拿到数字和最大的路径走,走k次,求最大和 8 9 这是 最大费用最大流 问题 每次spfa都

最小费用最大流问题

最小费用最大流问题 最小费用最大流问题: 在最大流有多组解时,给每条边在附上一个单位费用的量,问在满足最大流时的最小费用是多少? 来搞清楚一些概念: 最小费用最大流:指满足源点流出的流量最大且流量平衡时,总费用最小的一个网络. 最小费用可行流:指满足流量平衡时,总费用最小的一个网络. 最大费用最大流:把费用都取相反数的最小费用最大流,很容易证明等价. 最大费用可行流:把费用都取相反数的最小费用可行流,很容易证明等价. 思路 其实,就了解,解决最小费用最大流问题有两种思路: 一条途径是先用最大流算

图与网络优化——最小费用最大流问题

这个问题真是烧脑.不过弄懂了最后. 这个问题中构建的图是单位运价和容量的网络. 首先把费用提出来单独构建一个图,找到最短路经.然后用这条路径上的最下容量来调整这条路上的流量.调整过后,重新构图就是要把最短路的路径加上反向弧,利用公式当正向弧的时候流量小于容量则价钱不变,容量等于流量则价钱为无穷尽去掉.如果反向弧则流量大于0去价钱的相反数,若等于0则去无穷.调整之后从新找到最短路,然后用最小的容量进行调整,注意这个调整是差额调整,就是你的流量相对于上次变化了多少那相关的弧就要变化多少,不是容量是多

ACM/ICPC 之 卡卡的矩阵旅行-最小费用最大流(可做模板)(POJ3422)

将每个点拆分成原点A与伪点B,A->B有两条单向路(邻接表实现时需要建立一条反向的空边,并保证环路费用和为0),一条残留容量为1,费用为本身的负值(便于计算最短路),另一条残留容量+∞,费用为0(保证可以多次通过该点,但费用只计算一次). 另外伪点B与原点右侧与下方的点有一条单向路(邻接表实现需要建立反向空边),残留容量为+∞,费用为0.源点0到点1有一条单向路,残留容量为K(可以通过K次),最后一个点的伪点2*n*n与汇点2*n*n+1有一条单向边,残留容量为K,两条边的费用都为0. 构图成功

POJ2135 Farm Tour 【最小费用最大流】

Farm Tour Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 11782   Accepted: 4393 Description When FJ's friends visit him on the farm, he likes to show them around. His farm comprises N (1 <= N <= 1000) fields numbered 1..N, the first of

NWERC 2012 A - Admiral【最小费用最大流】

题意: 有n个岛屿 起点1 终点 n 现在两个人同时从起点出发向终点前进 两个人的路径不能有交集(一个人走过的路另一个人不能再走了) 问两个人的最小总花费 分析: 刚开始的的时候以为是最短路和次短路的和就行了   (太天真了) 快要结束的时候突然间发现这不就是传说中的最小费用最大流问题吗 可惜时间不够 思路是这样的:因为两个人来走,所以输入和输出的流量为1 费用为0 而连接的两点流量为1花费为权值 建图即可 结束之后敲完了代码,测试数据通过 不过没有地方提交 在程的帮助下重启pc^2完成了提交,

poj 2135 Farm Tour (最小费用最大流模板)

网络流的费用: 在实际应用中,与网络流有关的问题,不仅涉及流量,而且还有费用的因素.网络的每一条边(v,w)除了给定容量cap(v,w)外,还定义了一个单位流量费用cost(v,w) 最小费用最大流问题 给定网络G,要求G的一个最大用流flow,使流的总费用最小. 求解MCMF问题的算法: 最小费用最大流最常用和基本的算法我们可以称它为最小费用路算法,其思想与求最大流的增广路算法类似,不断在残流网络中寻找从源s到汇t的最小费用路,即残流网络中从s到t的以费用为权的最短路,然后沿最小费用路增流,直

网络流--最小费用最大流 (理解)

1.什么是最小费用最大流问题 上篇文章我们讲解了最大流问题,那什么是最小费用最大流呢?听名字就可以看出,我们要在满足最大流的同时找到达成最大流的最小费用. 对于一个网络流,最大流是一定的,但是组成最大流的费用是可以不同的,这里就有了在最大流网络上产生的费用流网络,就有了最小花费问题. 简单来说,就是满足最大流的路径可能有多条,我们要从这多条路径中找到一条花费代价最小的路径.所以最大流是解决这类问题的前提 2.EK算法 + SPFA 最短路   我们用每条边单位流量的花费作为边权,假如一条合法路径