UVA1659&HDU2982->Help Little Laura 帮助小萝拉 (循环费用流)

来源:互联网 发布:js form表单验证 编辑:程序博客网 时间:2024/04/29 21:25

题意:平面上有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\n",++cas,ans);    }    return 0;}

这里写图片描述
  

0 0
原创粉丝点击