有源汇上下界可行流(最大流/最小流)

以无源汇上下界可行流为基础,如果有源点汇点呢,源点汇点是不必平衡的,我们要转变成无源汇的,那么源点流出的就要有一个汇点流入源点,同理也需要汇点流出,那很简单了,从当前汇点引一条边到当前源点,容量为inf即可,然后跑无源汇有上下界可行流就好啦

这样跑出来一个后,如何调整到最大流 / 最小流呢?

求最大流应该好做一点,从s 到 t的残流网络,跑一跑最大流就好了,最后的可行流最大流量就是 : 最初可行流流量 + s - t残流最大流流量

稍微整理一下:
有源汇上下界可行流:
·根据题意建图

·汇点 到 源点 连入容量无穷大的边 转变为无源汇上下界可行流(循环流)

·添加附加流 跑Dinic

·得到基础可行流

·删除附加流的边,依照初始源点汇点和残流网络再跑DInic

·得到最后的有源汇上下界可行最大流 = 第一次Dinic后从初始汇点留到源点的流量 + 残流网络最大流的流量

·每条边的流量 = 最低限流  +  对应逆向边的流量

例题ZOJ3229

n天 m个人
每个人至少gx张
每天 有C个不同目标 且至多排 D张
C个目标 : 目标号 必须要拍l r张

依据输入建完图后,就是典型的有源汇上下界可行流了

先来看一下经典的初始化模板

#include <iostream>
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <queue>
#define inf (1 << 28)
using namespace std;
const int maxn = 2005,maxm=1e5+1e2;
//前向星
struct node{
    int to,val,lid,pre;
}e[maxm];
int id[maxn],cnt = 0;
//dinic
int cur[maxn];
int flor[maxn];//BFS的分层处理
int totid;
int n,m;//基础变量

int upflow[maxn];//差流(可升流)
int retf[maxm];//结果
int low[maxm];//下界

void init()
{
    memset(id,-1,sizeof(id));
    memset(upflow,0,sizeof(upflow));
    cnt = 0;
}
//网络流加边
void add(int from,int to,int val,int lid)
{
    e[cnt].lid = lid;
    e[cnt].to = to;
    e[cnt].val = val;
    e[cnt].pre = id[from];
    id[from] = cnt++;
    swap(from,to);
    e[cnt].lid = lid;
    e[cnt].to = to;
    e[cnt].val = 0;
    e[cnt].pre = id[from];
    id[from] = cnt++;
}
//Dinic
//bfs分层
bool bfs(int s,int t)
{
    memset(flor,0,sizeof(flor));
    flor[s] = 1;
    queue<int> q;
    while(q.size())q.pop();
    q.push(s);

    while(q.size())
    {
        int now = q.front();
        q.pop();

        for(int i = id[now];~i;i = e[i].pre)
        {
            int to = e[i].to;
            int val = e[i].val;
            if(val > 0 && flor[to] == 0)
            {
                flor[to] = flor[now] + 1;
                //printf("%d flor = %d\n",to,flor[to]);
                q.push(to);
                if(to == t)return 1;
            }
        }
    }
    return 0;
}
int dfs(int s,int t,int value)
{
    //printf("s t value = ::: %d %d %d\n",s,t,value);
    if(s == t || value == 0)return value;
    int ret = value,a;
    for(int &i = cur[s];~i;i = e[i].pre)
    {
        int to = e[i].to;
        int val = e[i].val;
        //printf("to = %d val = %d flornow = %d florto = %d\n",to,val,flor[s],flor[to]);
        if(flor[to] == flor[s] + 1 && (a = dfs(to,t,min(ret,val))))
        {
            //printf("a = %d\n",a);
            e[i].val -= a;
            e[i^1].val += a;
            ret -= a;
            if(ret == 0)break;
        }
    }
    if(ret == value)flor[s] = 0;

    return value - ret;
}
int dinic(int s,int t)
{
    int ret = 0;
    while(bfs(s,t))
    {
        memcpy(cur,id,sizeof(id));
        ret += dfs(s,t,inf);
        //cout<<ret<<endl;
    }
    return ret;
}

 对于已有的边我要记录流量差,以提供给附加流增流依据

void addflow(int from,int to,int low,int up,int lid)
{
    upflow[from] -= low;
    upflow[to] += low;
    add(from,to,up-low,lid);
}

 对于我们新增加的源点汇点就不用啦

剩下的操作我就一main到底了…………

根据题意建边,存储最低限流

init();
        s = 0;t = n + m + 1;
        ss = t + 1;
        tt = t + 2;
        totid = 0;
        for(int i = 1;i <= m;++i)
        {
            scanf("%d",&peoplelim);
            addflow(n+i,t,peoplelim,inf,0);
        }
        int targetnum,daylim;
        for(int i = 1;i <= n;++i)//天
        {
            scanf("%d%d",&targetnum,&daylim);
            addflow(s,i,0,daylim,0);
            for(int j = 1;j <= targetnum;j++)
            {
                int targetid,l,r;
                scanf("%d%d%d",&targetid,&l,&r);
                addflow(i,n+1+targetid,l,r,++totid);
                low[totid] = l;
            }
        }

 转变无源汇,求出基本可行流

int sum = 0;

        for(int i = s;i <= t;++i)
        {
            if(upflow[i] < 0)
                add(i,tt,-upflow[i],0);
            else
            {
                sum += upflow[i];
                add(ss,i,upflow[i],0);
            }
        }
        add(t,s,inf,0);

 存在基本可行流

删除添加的边,在原来的有源汇图中跑一遍残流网络最大流

if(dinic(ss,tt) == sum)
        {
//删边 for(int i = id[ss];~i;i = e[i].pre) { e[i].val = e[i^1].val = 0; } for(int i = id[tt];~i;i = e[i].pre) { e[i].val = e[i^1].val = 0; } int baseflow = e[cnt-1].val; e[cnt - 1].val = e[cnt - 2].val = 0; printf("%d\n",baseflow + dinic(s,t)); for(int now = 1;now <= m;++now) { for(int i = id[n+now];~i;i = e[i].pre) { int lid = e[i].lid; if(lid == 0)continue; if(i % 2 == 0)continue; retf[lid] = e[i].val + low[lid]; } } for(int i=1;i<=totid;++i)printf("%d\n",retf[i]); } else { printf("-1\n"); }

 完整代码:

/**
n天 m个人
每个人至少gx张
每天 有C个不同目标 且至多排 D张
C个目标 : 目标号 必须要拍l r张
**/

#include <iostream>
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <queue>
#define inf (1 << 28)
using namespace std;
const int maxn = 2005,maxm=1e5+1e2;
//前向星
struct node{
    int to,val,lid,pre;
}e[maxm];
int id[maxn],cnt = 0;
//dinic
int cur[maxn];
int flor[maxn];//BFS的分层处理
int totid;
int n,m;//基础变量

int upflow[maxn];//差流(可升流)
int retf[maxm];//结果
int low[maxm];//下界

void init()
{
    memset(id,-1,sizeof(id));
    memset(upflow,0,sizeof(upflow));
    cnt = 0;
}
//网络流加边
void add(int from,int to,int val,int lid)
{
    e[cnt].lid = lid;
    e[cnt].to = to;
    e[cnt].val = val;
    e[cnt].pre = id[from];
    id[from] = cnt++;
    swap(from,to);
    e[cnt].lid = lid;
    e[cnt].to = to;
    e[cnt].val = 0;
    e[cnt].pre = id[from];
    id[from] = cnt++;
}
//Dinic
//bfs分层
bool bfs(int s,int t)
{
    memset(flor,0,sizeof(flor));
    flor[s] = 1;
    queue<int> q;
    while(q.size())q.pop();
    q.push(s);

    while(q.size())
    {
        int now = q.front();
        q.pop();

        for(int i = id[now];~i;i = e[i].pre)
        {
            int to = e[i].to;
            int val = e[i].val;
            if(val > 0 && flor[to] == 0)
            {
                flor[to] = flor[now] + 1;
                //printf("%d flor = %d\n",to,flor[to]);
                q.push(to);
                if(to == t)return 1;
            }
        }
    }
    return 0;
}
int dfs(int s,int t,int value)
{
    //printf("s t value = ::: %d %d %d\n",s,t,value);
    if(s == t || value == 0)return value;
    int ret = value,a;
    for(int &i = cur[s];~i;i = e[i].pre)
    {
        int to = e[i].to;
        int val = e[i].val;
        //printf("to = %d val = %d flornow = %d florto = %d\n",to,val,flor[s],flor[to]);
        if(flor[to] == flor[s] + 1 && (a = dfs(to,t,min(ret,val))))
        {
            //printf("a = %d\n",a);
            e[i].val -= a;
            e[i^1].val += a;
            ret -= a;
            if(ret == 0)break;
        }
    }
    if(ret == value)flor[s] = 0;

    return value - ret;
}
int dinic(int s,int t)
{
    int ret = 0;
    while(bfs(s,t))
    {
        memcpy(cur,id,sizeof(id));
        ret += dfs(s,t,inf);
        //cout<<ret<<endl;
    }
    return ret;
}


void addflow(int from,int to,int low,int up,int lid)
{
    upflow[from] -= low;
    upflow[to] += low;
    add(from,to,up-low,lid);
}
int main()
{
    int s,t;//初始源汇
    int ss,tt;//无源汇有上下界可行流种附加流的源汇
    int peoplelim;//每个人至少拍的张数

    while(~scanf("%d%d",&n,&m))
    {
        init();
        s = 0;t = n + m + 1;
        ss = t + 1;
        tt = t + 2;
        totid = 0;
        for(int i = 1;i <= m;++i)
        {
            scanf("%d",&peoplelim);
            addflow(n+i,t,peoplelim,inf,0);
        }
        int targetnum,daylim;
        for(int i = 1;i <= n;++i)//天
        {
            scanf("%d%d",&targetnum,&daylim);
            addflow(s,i,0,daylim,0);
            for(int j = 1;j <= targetnum;j++)
            {
                int targetid,l,r;
                scanf("%d%d%d",&targetid,&l,&r);
                addflow(i,n+1+targetid,l,r,++totid);
                low[totid] = l;
            }
        }
        //bound_flow();//有源汇上下界可行最大流
        int sum = 0;

        for(int i = s;i <= t;++i)
        {
            if(upflow[i] < 0)
                add(i,tt,-upflow[i],0);
            else
            {
                sum += upflow[i];
                add(ss,i,upflow[i],0);
            }
        }
        add(t,s,inf,0);
        if(dinic(ss,tt) == sum)
        {
            for(int i = id[ss];~i;i = e[i].pre)
            {
                e[i].val = e[i^1].val = 0;
            }
            for(int i = id[tt];~i;i = e[i].pre)
            {
                e[i].val = e[i^1].val = 0;
            }
            int baseflow = e[cnt-1].val;
            e[cnt - 1].val = e[cnt - 2].val = 0;
            printf("%d\n",baseflow + dinic(s,t));
            for(int now = 1;now <= m;++now)
            {
                for(int i = id[n+now];~i;i = e[i].pre)
                {
                    int lid = e[i].lid;
                    if(lid == 0)continue;
                    if(i % 2 == 0)continue;
                    retf[lid] = e[i].val + low[lid];
                }
            }
            for(int i=1;i<=totid;++i)printf("%d\n",retf[i]);
        }
        else
        {
            printf("-1\n");
        }
        printf("\n");
    }
}

 

posted @ 2018-09-30 22:37  Butterflier  阅读(998)  评论(0编辑  收藏  举报