zoukankan      html  css  js  c++  java
  • UVA1659&HDU2982->Help Little Laura 帮助小萝拉 (循环费用流)

    题意:平面上有m条有向线段连接了n个点。
    你从某个点出发顺着有向线段行走,给走过的每条线段涂一种不同的颜色,最后回到起点。
    你可以多次行走,给多个回路涂色(要么不涂色,要么就至少给一个回路上的边全部涂色)。
    可以重复经过一个点,但不能重复经过一条有向线段。
    如下图所示的是一种涂色方法(虚线表示未涂色,即每次都可以从任意点出发染色)。
    每涂一个单位长度将得到X分,但每使用一种颜色将扣掉Y分。
    假设你拥有无限多种的颜色,问如何涂色才能使得分最大?
    输入保证若存在有向线段u -> v,则不会出现有向线段v -> u。
    n <= 100,m <= 500,1 <= X,Y <= 1000。
    对于坐标(x,y)0 <= x,y <= 1000。
    这里写图片描述

    思路:看刘汝佳的方法,还没有深入理解。
    http://blog.csdn.net/u013368721/article/details/30553815
    http://www.cnblogs.com/xcw0754/p/4659201.html

      要求的就是最大费用循环流(即每找到一个环就可以进行增广)。找环可能并不复杂,但是要找一个最大的环就有点复杂了,所以用网络流解决。又因为找的是最大费用,按老套路的话会出现无限增大费用的情况,所以要先将每条边的费用取相反数(前面加个负),才可以有机会求最小费用流。而这些边的权有正有负,取完之后也可能出现负环了,所以主要问题就是解决负环。
      用最小费用流求最大费用循环流时,解决负环的一种方法:
    (1)先将所有边权取反。
    (2)建边。正权值的边容量为1,费用为权值。负权值的边u->v拆成3条边,
    分别是S->v,v->u,u->T,容量都为1,v->u费用为负权的相反数,其他2条费用为0。
    这样会出现某个点有多条边连到S或T,可以互相抵消到一方为0为止,统计剩下多少条k,将其中1条的容量设为k,其他的全部删掉。如果全部抵消掉了,那就将连S和T的边全部删掉。(这个删边的方法有技巧)
    (3)跑一次最小费用流得到的总费用,加上所有负权之和之后(注:此时答案已为负的),再取反即得到最大费用。

      删边技巧是,在建这S->v,v->u,u->T 三条边时,先建中间那条,统计该点连到S几次,减去连到T点几次,结果若为正,则与S连一条边,容量就是几次,若负,同理。

      至于why it works!得好好想想~
      画几个点验证了一下发现,如果一个原图中的环(权值大于0)值得取,那么流会自动流向该环原图中的负权边。而如果不值得取,那么会流向原图中的正权边。因为我们是用sum(负值)加上那个费用(正值),所以当该环要取时,则自动减去那些负权,不取呢,会自动减去那些正权(而那些负权的完全没取到)。不懂就画个环出来验证吧。

    ===========================
    一开始用lrj的vector的方法存边,TLE了一发
    改成前向星存图,过了,STL还是慢啊

    #include<queue>
    #include<cmath>
    #include<cstdio>
    #include<vector>
    #include<cstring>
    #include<iostream>
    #include<algorithm>
    using namespace std;
    typedef long long LL;
    const int N = 2007;
    const int INF=0x3f3f3f3f;
    const double EPS = 1e-6;
    struct MCMF{
        struct Edge{
            int from,to,cap,flow,nxt;
            double cost;
            Edge(){}
            Edge(int x,int y,int z,int u,double v,int n){
                from=x;to=y;cap=z;flow=u;cost=v;nxt=n;
            }
        }edges[N];
        int E,head[N];
        int n,s,t,inq[N],p[N],a[N];
        double d[N];
    
        inline void Init(int n,int s,int t){
            this->n = n; E = -1;
            this->s = s; this->t = t;
            memset(head,-1,sizeof(head));
        }
        inline void AddEdge(int f,int t,int c,double w){
            edges[++E] = Edge(f,t,c,0, w,head[f]);
            head[f] = E;
            edges[++E] = Edge(t,f,0,0,-w,head[t]);
            head[t] = E;
        }
    
        bool spfa(int s,int t,int flow,double &cost){
            for (int i=0;i<=n;i++)d[i]=INF;
            memset(inq,0,sizeof(inq));
            d[s]=0;inq[s]=1;p[s]=0;a[s]=INF;
            queue<int>Q;Q.push(s);
            for (;!Q.empty();){
                int nxt, u =Q.front();Q.pop();inq[u]=0;
                for (int i=head[u];i!=-1;i=nxt){
                    Edge &e = edges[i]; nxt = e.nxt;
                    if (e.cap<=e.flow||d[e.to]<=d[u]+e.cost)continue;
                    d[e.to] = d[u] + e.cost;
                    p[e.to] = i;
                    a[e.to] = min(a[u],e.cap-e.flow);
                    if (!inq[e.to]){Q.push(e.to);inq[e.to]=1;}
                }
            }
            if (d[t]==INF)return 0;//false
            flow += a[t];
            cost += (double)d[t]*(double)a[t];
            for (int u=t;u!=s;u=edges[p[u]].from){
                edges[p[u]  ].flow += a[t];
                edges[p[u]^1].flow -= a[t];
            }
            return 1;//true
        }
    
        //需要保证初始网络中没有负权
        double mcmf(){
            int flow =0;
            double cost = 0;
            for (;spfa(s,t,flow,cost););
            return cost;
        }//MinCostMaxFlow
    } g ;
    
    struct point{
        int x,y,d;//d:Degree
        point(){}
        inline void read(){scanf("%d%d",&x,&y);d=0;}
    }po[N];
    vector <int > link[N];
    inline double sqr(double x){return x*x; }
    inline double dist(int a,int b){
        return sqrt(sqr(po[a].x-po[b].x)+ sqr(po[a].y-po[b].y));
    }
    
    int main(){
        //freopen("in.txt","r",stdin);
        int n,dx,dy,x;
        for (int cas=0;~scanf("%d",&n)&&n;){
            scanf("%d%d",&dx,&dy);
            g.Init(n+1,0,n+1);
            for (int i=1;i<=n;i++)link[i].clear();
            for (int i=1;i<=n;i++){
                po[i].read();
                for (;~scanf("%d",&x)&&x;)link[i].push_back(x);
            }
            double ans = 0;
            for (int i=1;i<=n;i++){
                for (int j=0;j<link[i].size();j++){
                    double d = dist(i,link[i][j]);
                    double cost = (double)dy - 1.0*d * dx ;
                    if (cost>0) g.AddEdge(i,link[i][j],1,cost);
                    else {
                        ans -= cost;
                        g.AddEdge(link[i][j],i,1,-cost);
                        po[link[i][j]].d++;
                        po[i].d--;
                    }
                }
            }
            for (int i=1;i<=n;i++){
                if (po[i].d>0) g.AddEdge(g.s, i, po[i].d, 0);
                if (po[i].d<0) g.AddEdge(i, g.t,-po[i].d, 0);
            }
            ans -= g.mcmf()-EPS;
            printf("Case %d: %.2lf
    ",++cas,ans);
        }
        return 0;
    }
    

    这里写图片描述
      

  • 相关阅读:
    MySQL索引背后的数据结构及算法原理 [转]
    5.5下对DDL操作提速的测试
    由浅入深理解索引的实现(2) [转]
    由浅入深理解索引的实现(1) [转]
    两个比较有用的字符串函数
    在慢查询里保留注释部分
    想在Innodb表上做OPTIMIZE操作?先等等看再说!
    Win CE和smartphone和pocket pc和windows mobile比较(zt)
    学习笔记(配置SQL Server 2005允许远程连接)
    配置程序集的版本策略(zt)
  • 原文地址:https://www.cnblogs.com/cww97/p/12349368.html
Copyright © 2011-2022 走看看