zoukankan      html  css  js  c++  java
  • poj2125 最小点权覆盖集

     题意:有一张图,对于每个点,有出边和入边,现在目的是删除改图的所有边,对于每个点,删除出边的花费Wi-,删除入边的花费Wi+,现在的目的求删去所有边后的花费最小。

    建图方法:对于每个点i,拆点为i,i+n,对于入边,从i+n想汇点T连边,值为入边花费;对于出边,从S向i连边,权值为出边花费。m组相连的边(x,y),从x向y+n连边, 费用INF。求出最小割。

    然后是计算删除的是哪些点。对于完成最小割后的图,从源点S进行dfs,如果能够访问到,标记。对于不能访问到的点,i<=n时,这个点的出边是属于割集的,即所求点,且为‘-’;如果i>n,表示该点为需要删除的边的点,为'+';

    最小割后的图片

    (ps:S,T写错了。。。)

    根据图很明显的可以看出哪些边时需要删除,并且是哪些点的边。

    #include<stdio.h>
    #include<string.h>
    #include<queue>
    #define INF 99999999
    using namespace std;
    const int maxn = 120*2;
    const int maxm = 5010;
    struct node
    {
        int to;
        int v;
        int flag;
        int next;
    }edge[(maxn+maxm)*2];
    struct ans_point
    {
        int x;
        int flag;
    }ans_p[maxn];
    int index,pre[maxn],vis[maxn],S,T;
    int in[maxn],out[maxn];
    void add(int x,int y,int z)
    {
        edge[index].to=y;
        edge[index].v=z;
        edge[index].flag=index+1;
        edge[index].next=pre[x];
        pre[x]=index++;
        edge[index].to=x;
        edge[index].v=0;
        edge[index].flag=index-1;
        edge[index].next=pre[y];
        pre[y]=index++;
    }
    int dfs(int u,int low)
    {
        int i,used=0;
        if(u==T)
            return low;
        for(i=pre[u];i!=-1&&used<low;i=edge[i].next)
        {
            if(edge[i].v&&vis[edge[i].to]==vis[u]+1)
            {
                int a=dfs(edge[i].to,min(low-used,edge[i].v));
                edge[i].v-=a;
                edge[edge[i].flag].v+=a;
                used+=a;
            }
        }
        if(!used)
            vis[u]=-1;
        return used;
    }
    bool BFS()
    {
        int i;
        queue<int>q;
        memset(vis,-1,sizeof(vis));
        vis[0]=1;
        q.push(0);
        while(!q.empty())
        {
            int t=q.front();
            q.pop();
            for(i=pre[t];i!=-1;i=edge[i].next)
            {
                if(edge[i].v&&vis[edge[i].to]<0)
                {
                    vis[edge[i].to]=vis[t]+1;
                    q.push(edge[i].to);
                }
            }
        }
        if(vis[T]>0)
            return true;
        return false;
    }
    void cnt_dfs(int u)
    {
        int i;
        vis[u]=1;
        for(i=pre[u];i!=-1;i=edge[i].next)
        {
            if(edge[i].v>0&&!vis[edge[i].to])
                cnt_dfs(edge[i].to);
        }
    }
    int main()
    {
        int n,m,i,j;
        while(~scanf("%d%d",&n,&m))
        {
            index=1;
            memset(pre,-1,sizeof(pre));
            S=0,T=2*n+1;
            for(i=1;i<=n;i++)
            {
                scanf("%d",&in[i]);
                add(i+n,T,in[i]);
            }
            for(i=1;i<=n;i++)
            {
                scanf("%d",&out[i]);
                add(S,i,out[i]);
            }
            while(m--)
            {
                int x,y;
                scanf("%d%d",&x,&y);
                add(x,y+n,INF);
            }
            int ans=0;
            while(BFS())
            {
                int a=dfs(0,INF);
                if(!a)break;
                ans+=a;
            }
            printf("%d
    ",ans);
            memset(vis,0,sizeof(vis));
            cnt_dfs(0);
            int cnt=0;
            for(i=1;i<=n*2;i++)
            {
                if(i>n)
                {
                    if(vis[i])
                    {
                        ans_p[cnt].x=i;
                        ans_p[cnt++].flag=1;
                    }
                }
                else
                {
                    if(!vis[i])
                    {
                        ans_p[cnt].x=i;
                        ans_p[cnt++].flag=0;
                    }
                }
            }
            printf("%d
    ",cnt);
            for(i=0;i<cnt;i++)
            {
                if(ans_p[i].x>n)
                        printf("%d ",ans_p[i].x-n);
                else printf("%d ",ans_p[i].x);
                if(ans_p[i].flag==0)
                    printf("-
    ");
                else printf("+
    ");
            }
        }
    }
  • 相关阅读:
    Fast exit from dram self-refresh
    关于Net开发中一些SQLServer性能优化的建议
    收集一些优秀的DoNet开源项目
    收集一些优秀的DoNet开源项目
    收集一些优秀的DoNet开源项目
    LINQ表达式用法整理
    LINQ表达式用法整理
    LINQ表达式用法整理
    SQL 拼接多个字段的值&一个字段多条记录的拼接
    你应该知道的jQuery技巧【收藏】
  • 原文地址:https://www.cnblogs.com/sweat123/p/4924380.html
Copyright © 2011-2022 走看看