• nyoj-1367-河南省第十一届省赛-E物流配送-最小费用流


    1367-物流配送


    内存限制:128MB 时间限制:8000ms 特判: No
    通过数:1 提交数:1 难度:4

    题目描述:

    物流配送是物流活动中一种非单一的业务形式,它与物品流动、资金流动紧密结合。备货是配送的准备工作或基础工作,备货工作包括筹集货源、订货或购货、集货、进货及有关的质量检查、结算、交接等。配送的优势之一,就是可以集中用户的需求进行一定规模的备货。备货是决定配送成败的初期工作,如果备货成本太高,会大大降低配送的效益。配送中的储存有储备及暂存两种形态。配送储备是按一定时期的配送经营要求,形成的对配送的资源保证。这种类型的储备数量较大,储备结构也较完善,视货源及到货情况,可以有计划地确定周转储备及保险储备结构及数量。

    Dr. Kong 所在的研究团队准备为Hai-E集团开发一个物流配送管理系统。已知Hai-E集团已经在全国各地建立了n个货物仓库基地,任意两个基地的货物可以相互调配。现在需要根据用户订货要求,来重新调配每个基地的货物数量。为了节流开源,希望对整个物流配送体系实行统一的货物管理和调度,能够提供一个全面完善的物流仓储配送解决方案,以减少物流配送过程中成本、人力、时间。

    输入描述:

    第一行:   n             (1 ≤ n ≤ 1000)
    
    第2行:   a1  a2 …… an    表示n个基地当前的物品数量             (0≤ ai ≤ 106 )
    
    第3行:   b1  b2 …… bn   表示调配后,每个基地i应不少于bi个物品  (0≤ bi ≤ 106)
    
    接下来n-1行,每行三个整数: i  j  k 表示从第i基地调配一个物品到第j基地需要花费为k,或 从第j基地调配一个物品到第i基地需要花费为k。(0≤ k ≤ 10^6)

    输出描述:

    输出配送后的最小费用。
    
    已知: a1+a2+…+an >=b1+b2+…+bn

    样例输入:

    6
    0 1 2 2 0 0
    0 0 1 1 1 1
    1 2 2    
    1 3 5
    1 4 1
    2 5 5
    2 6 1

    样例输出:

    9

    提示:

        是个裸题把算是,当时没学不会,然后学了最大流之后,看了眼费用流的含义,对着题目yy着

    建了个模,套了套模板竟然A了= =不知道是不是数据太水。

        按照题目的意思一定存在一种方案使得满足所有条件,所以不必考虑不满流的情况。

        建立一个源点S一个汇点T,S连向所有的点,容量是a[i]费用是0,表示每个地方可以免费获得

    a[i]个物品,这与每个点初始有a[i]个物品是等价的。每个点向T连边,容量是b[i]费用是0,表示每个点

    可以免费移到T b[i]个物品,(因为要求的是最小的费用,所以最大流显然设置为∑bi是最优的),然后跑

    费用流就好了。

      

     1 #include<bits/stdc++.h>
     2 using namespace std;
     3 #define LL long long 
     4 #define inf 0x3f3f3f3f
     5 #define pb push_back
     6 #define pii pair<int,int>
     7 struct Edge{
     8   int v,cap,flow,cost,next;
     9 }e[100010];
    10 int first[1010],a[1010],b[1010],d[1010],f[1010],p[1010][2],tot,N,S,T;
    11 bool inq[1010];
    12 void add(int u,int v,int cap,int flow,int cost){
    13     e[tot]=Edge{v,cap,flow,cost,first[u]};
    14     first[u]=tot++;
    15 }
    16 bool spfa(int &flow,int &cost){
    17   memset(d,inf,sizeof(d));
    18   memset(inq,0,sizeof(inq));
    19   d[S]=0,inq[S]=1,p[S][0]=p[S][1]=0,f[S]=inf;
    20   queue<int>q;
    21   q.push(S);
    22   while(!q.empty()){
    23     int u=q.front();q.pop();
    24     inq[u]=0;
    25     for(int i=first[u];~i;i=e[i].next){
    26       if(e[i].cap>e[i].flow && d[e[i].v]>d[u]+e[i].cost){
    27         d[e[i].v]=d[u]+e[i].cost;
    28         p[e[i].v][0]=i;
    29         p[e[i].v][1]=u;
    30         f[e[i].v]=min(f[u],e[i].cap-e[i].flow);
    31         if(!inq[e[i].v])q.push(e[i].v);
    32       }
    33     }
    34   }
    35 
    36   if(d[T]==inf) return false;
    37   flow+=f[T];
    38   cost+=d[T]*f[T];
    39   int u=T;
    40   while(u!=S){
    41     e[p[u][0]].flow+=f[T];
    42     e[p[u][0]^1].flow-=f[T];
    43     u=p[u][1];
    44   }
    45   return true;
    46 }
    47 void solve(){
    48   int flow=0,cost=0;
    49   while(spfa(flow,cost));
    50   cout<<cost<<endl;
    51 }
    52 int main(){
    53   while(scanf("%d",&N)==1){
    54     int u,v,w;
    55     tot=0,memset(first,-1,sizeof(first));
    56     S=N+1,T=N+2;
    57     for(int i=1;i<=N;++i){
    58       scanf("%d",a+i);
    59       add(S,i,a[i],0,0);
    60       add(i,S,0,0,0);
    61     }
    62     for(int i=1;i<=N;++i){
    63       scanf("%d",b+i);
    64       add(i,T,b[i],0,0);
    65       add(T,i,0,0,0);
    66     }
    67     for(int i=1;i<N;++i){
    68       scanf("%d%d%d",&u,&v,&w);
    69       add(u,v,inf,0,w),add(v,u,0,0,-w);
    70       add(v,u,inf,0,w),add(u,v,0,0,-w);
    71     }
    72     solve();
    73   }
    74   return 0;
    75 }
  • 相关阅读:
    java基础
    php中的$_REQUEST超全局变量
    update 数据表 set 字段1=字段1+id的sql语句
    青蛙跳台阶的相关问题
    Java语言实现石头剪刀布游戏
    Java语言实现palindrome(回文)
    Java语言实现奇怪的比赛
    Java语言实现随意组合
    Java编辑器IDEA的下载与安装
    Vscode下载与汉化
  • 原文地址:https://www.cnblogs.com/zzqc/p/9395771.html
Copyright © 2020-2023  润新知