• 【zkw费用流】[网络流24题]餐巾计划问题


    题目描述

    一个餐厅在相继的N天里,第i天需要Ri块餐巾(i=l,2,…,N)。餐厅可以从三种途径获得餐巾。

    (1)购买新的餐巾,每块需p分;

    (2)把用过的餐巾送到快洗部,洗一块需m天,费用需f分(f<p)。如m=l时,第一天送到快洗部的餐巾第二天就可以使用了,送慢洗的情况也如此。

    (3)把餐巾送到慢洗部,洗一块需n天(n>m),费用需s分(s<f)。

    在每天结束时,餐厅必须决定多少块用过的餐巾送到快洗部,多少块送慢洗部。在每天开始时,餐厅必须决定是否购买新餐巾及多少,使洗好的和新购的餐巾之和满足当天的需求量Ri,并使N天总的费用最小

    输入输出格式

    输入格式:

    输入文件共3行,第1行为总天数;第2行为每天所需的餐巾块数;第3行为每块餐巾的新购费用p,快洗所需天数m,快洗所需费用f,慢洗所需天数n,慢洗所需费用s。

    输出格式:

    输出文件共1行为最小的费用。

    输入输出样例

    输入样例#1: 
    3
    3 2 4
    10 1 6 2 3
    
    输出样例#1: 
    64
    

    题解

    这题的建模真的不好想。。。

    首先对于所有天数拆点,拆成X集合和Y集合

    其中Xi表示第i天用完的餐巾,Yi表示第i天需要的餐巾

      从s到每个Xi连一条流量为Ri,花费为0的边,从每个Yi到t也连一条流量为Ri,花费为0的边

      从Xi到Xi+1连一条流量为inf,花费为0的边,表示Xi天多出没用的可以直接给Xi+1天

      从Xi到Yi+fast_time连一条流量为inf,花费为fast_cost的边,表示快洗

      从Xi到Yi+slow_time连一条流量为inf,花费为slow_cost的边,表示慢洗

    之后跑zkw费用流就行

    %%%zkw

    代码

    //by 减维
    #include<cstdio>
    #include<iostream>
    #include<cstring>
    #include<queue>
    #include<cstdlib>
    #include<ctime>
    #include<cmath>
    #include<map>
    #include<bitset>
    #include<algorithm>
    #define ll long long
    #define maxn 4005
    #define inf 1<<30
    using namespace std;
    
    struct edge{
        int to,ne,cap,val;
    }e[maxn<<4];
    
    int n,s,t,ft,fc,st,sc,cost,ecnt=1,head[maxn],dis[maxn];
    ll ans;
    bool pd[maxn],vis[maxn];
    deque<int>q;
    
    void add(int x,int y,int z,int k){++ecnt;e[ecnt]=(edge){y,head[x],z,k};head[x]=ecnt;}
    void addd(int x,int y,int z,int k){add(x,y,z,k);add(y,x,0,-k);}
    
    bool bfs()
    {
        memset(pd,0,sizeof(pd));
        for(int i=0;i<=t;++i)dis[i]=inf;
        pd[t]=1;dis[t]=0;q.push_back(t);
        while(!q.empty())
        {
            int d=q.front();q.pop_front();
            pd[d]=0;
            for(int i=head[d];i;i=e[i].ne)
            {
                int dd=e[i].to;
                if(e[i^1].cap&&dis[dd]>dis[d]-e[i].val)
                {
                    dis[dd]=dis[d]-e[i].val;
                    if(!pd[dd]){
                        pd[dd]=1;
                        if(q.empty()||dis[dd]>dis[q.front()])q.push_back(dd);
                        else q.push_front(dd);
                    }
                }
            }
        }
        return dis[s]<inf;
    }
    
    int dfs(int x,int cap)
    {
        vis[x]=1;
        if(x==t||cap==0)return cap;
        int tmp,ret=0;
        for(int i=head[x];i;i=e[i].ne)
        {
            int dd=e[i].to;
            if((!vis[dd])&&e[i].cap&&dis[dd]==dis[x]-e[i].val)
            {
                tmp=dfs(dd,min(e[i].cap,cap));
                cap-=tmp;ret+=tmp;
                e[i].cap-=tmp;e[i^1].cap+=tmp;
            }
        }
        return ret;
    }
    
    int main()
    {
        scanf("%d",&n);
        s=0,t=2*n+2;
        for(int i=1,x;i<=n;++i)
        {
            scanf("%d",&x);
            addd(0,i,x,0);
            addd(i+n,t,x,0);
        }
        scanf("%d%d%d%d%d",&cost,&ft,&fc,&st,&sc);
        for(int i=1;i<=n;++i)
        {
            if(i+1<=n)addd(i,i+1,inf,0);
            if(i+ft<=n)addd(i,i+n+ft,inf,fc);
            if(i+st<=n)addd(i,i+n+st,inf,sc);
            addd(0,i+n,inf,cost);
        }
        while(bfs())
        {
            vis[t]=1;
            while(vis[t])
            {
                memset(vis,0,sizeof(vis));
                int tmp=dfs(s,inf);
                ans+=(ll)tmp*dis[s];
            }
        }
        printf("%lld",ans);
        return 0;
    }
  • 相关阅读:
    Sa提权sql语句大全
    转载 使用AJAX实现动态刷新页面功能
    VirtualBox宿主机与虚拟机的数据共享数据空间
    SP_OACreate提权经验
    linux mysql启动错误
    Microsoft SQL Server SA权限最新入侵方法 不用net user添加账户
    Java中关于HashMap的使用和遍历
    转载 突破360防黑加固添加用户
    个人随想BLOG
    面试题收集
  • 原文地址:https://www.cnblogs.com/rir1715/p/8024648.html
Copyright © 2020-2023  润新知