zoukankan      html  css  js  c++  java
  • hdu 3667 /2010哈尔滨赛区H题 费用与流量为非线性关系/费用流

    题意: 在一般费用流题目改动:路过某路,每x单位流量须要花费 ai*x^2(ai为给定的系数)。

    開始的的时候,一看仅仅只是是最后统计费用上在改动罢了,一看例子。发现根本没那么简单(ps:以后每次写程序前先看例子能不能过。),由于是成平方关系。每次一流量增广的话。下次未必最优!于是想到我每次仅仅增长一个单位流量。每次增长一个单位之后,该路径上的全部边的费用边改为i^2-(i-1)^2,(第一次肯定增广a,其次的话3a.5a.7a.。。。由于第一次已经增广了,和为第i次i平方就可以!

    )如此。符合题意!

    一提交Wa!扫了一遍代码,发现反向边的费用没有改!

    于是改动:想到反向边的本质是提供懊悔机会,而我控制的是每次增长一个单位!懊悔一次的话,必定是付上一次的费用!

    所以,与平时不同:初始化反向边费用也为W(不是-W)。每次增广改动为-1a.-3a.-5a,这样控制比正向边慢一步。(事实上这才是反向边的本质作用,仅仅是平时时候费用是不变的,所以无需如此!)提交后AC!

    (ps:感觉对费用流算法内部逻辑更加清晰了有木有)

    后来看了网上的题解:看数据流量为<5,所以每条边拆为流量为1的,每条费用也如此递增。其次,这样本质是还是每次增广流量1,可谓异曲同工。

    只是拆边的思想值得借鉴。

    额外收获:1:注意看数据范围,有时候能够从上面思考算法。2:写程序前先过一遍例子,也许有思路启示。


    #include<cstdio>
    #include<iostream>
    #include<queue>
    #include<cstring>
    using namespace std;
    const int maxv=200;
    const int maxe=5000*2*2;
    const int inf=0x3f3f3f3f;
    int nume=0;int e[maxe][5];int head[maxv];
    int n,m,k;
    void inline adde(int i,int j,int c,int w,int f)         //新加入一个參数f,记录初始费用。费用w改变
    {
        e[nume][0]=j;e[nume][1]=head[i];head[i]=nume;
        e[nume][2]=c;e[nume][3]=w;e[nume++][4]=f;
        e[nume][0]=i;e[nume][1]=head[j];head[j]=nume;
        e[nume][2]=0;e[nume][3]=w;e[nume++][4]=f;
    }
    int inq[maxv];int pre[maxv];int prv[maxv];
    int d[maxv];
    bool spfa(int &sum,int &flow)
    {
        int s=0,t=n+1;
        for(int i=0;i<=n+3;i++)
              {
                  inq[i]=0;
                  d[i]=inf;
              }
        queue<int>q;
        q.push(s);
        inq[s]=1;
        d[s]=0;
        while(!q.empty())
        {
            int cur=q.front();
            q.pop();
            inq[cur]=0;
            for(int i=head[cur];i!=-1;i=e[i][1])
            {
                int v=e[i][0];
                if(e[i][2]>0&&d[cur]+e[i][3]<d[v])                 
                {
                    d[v]=d[cur]+e[i][3];
                    pre[v]=i;
                    prv[v]=cur;
                    if(!inq[v])
                    {
                        q.push(v);
                        inq[v]=1;
                    }
                }
            }
        }
        if(d[t]==inf)return 0;
        int cur=t;
        int minf=1;                              //每次增广一个单位
        while(cur!=s)
        {
            int fe=pre[cur];
            e[fe][3]+=e[fe][4]*2;                //关键点。
            e[fe^1][3]-=e[fe][4]*2;              //比正向边慢一步更新。(初始值来控制)
            cur=prv[cur];
        }
         cur=t;
        while(cur!=s)
        {
            e[pre[cur]][2]-=minf;
            e[pre[cur]^1][2]+=minf;
            cur=prv[cur];
        }
        flow+=minf;
        sum+=d[t];                       //,每次一单位,不用说,最短路即为总费用
        return 1;
    }
    int mincost(int &flow)
    {
        int sum=0;
        while(spfa(sum,flow));
        return sum;
    }
    void init()
    {
        nume=0;
        for(int i=0;i<=n+2;i++)
           head[i]=-1;
    }
    int main()
    {
        while(~scanf("%d%d%d",&n,&m,&k))
        {
            init();
             int a,b,x,c;
                for(int j=0;j<m;j++)
                 {
                     scanf("%d%d%d%d",&a,&b,&x,&c);
                     adde(a,b,c,x,x);
                 }
                adde(0,1,k,0,0);
                adde(n,n+1,k,0,0);                    //附加边来推断流量满否,方便
                int flow=0;
            int ans=mincost(flow);
            if(flow!=k)
              printf("-1
    ");
            else printf("%d
    ",ans);
        }
        return 0;
    }
    



  • 相关阅读:
    千个常用DOS命令全面收藏
    面向对象设计的11原则
    SQL语句判断指定的数据库、表、字段、存储过程是否存在
    ASP.NET MVC2 Areas区域新概念
    标准的 SQL 解析顺序
    Improvements to workspaces in TFS 2010
    jquery ajax return值不能取得的解决方案
    用 SQL 语句创建数据库用户(SQL Server 2005)
    简单实现.net MVC自定义错误处理页面
    自定义截图类(C#)
  • 原文地址:https://www.cnblogs.com/yfceshi/p/7258209.html
Copyright © 2011-2022 走看看