zoukankan      html  css  js  c++  java
  • poj3422 最小费用流

     一遍的话秩序要dp就好,但是这里要删去点。此题可以转化为最小费用流。开始我想了半天纠结怎么处理到过一次后值变0,看了书之后发现拆点解决了这个问题。

    对于点t,拆为t-->t',容量为1,费用为负的矩阵的值,那就解决了只能一次。如果2个点连通,s-->t,就相连,s-->t,s-->t',t'->s',t'->s,容量为无穷,费用0。

    然后费用流,得到的值取负即可。

    #include<stdio.h>
    #include<string.h>
    #include<queue>
    #define INF 99999999
    using namespace std;
    const int maxn = 5560;
    struct mp
    {
        int v;
        int cnt;
    } map[55][55];
    struct node
    {
        int to;
        int v;
        int cost;
        int flag;
        int next;
    } edge[maxn*maxn/2];
    int pre[maxn],fpre[maxn],index,head[maxn],dis[maxn],n,k,N;
    int ok(int x1,int y1,int x2,int y2)
    {
        if(x1>=n||y1>=n||x2>=n||y2>=n)
            return 0;
        return 1;
    }
    void add(int x,int y,int v,int cost)
    {
        edge[index].to=y;
        edge[index].v=v;
        edge[index].cost=cost;
        edge[index].flag=index+1;
        edge[index].next=head[x];
        head[x]=index++;
        edge[index].to=x;
        edge[index].v=0;
        edge[index].cost=-cost;
        edge[index].flag=index-1;
        edge[index].next=head[y];
        head[y]=index++;
    
    }
    int spfa(int S,int T)
    {
        int vis[maxn],i;
        queue<int>q;
        memset(vis,0,sizeof(vis));
        memset(pre,-1,sizeof(pre));
        for(i=0; i<=T; i++)
            dis[i]=INF;
        dis[S]=0;
        vis[S]=1;
        pre[S]=0;
        q.push(S);
        while(!q.empty())
        {
            int t=q.front();
            q.pop();
            vis[t]=0;
            for(i=head[t]; i!=-1; i=edge[i].next)
            {
                if(edge[i].v&&dis[edge[i].to]>dis[t]+edge[i].cost)
                {
                    dis[edge[i].to]=dis[t]+edge[i].cost;
                    if(!vis[edge[i].to])
                    {
                        vis[edge[i].to]=1;
                        q.push(edge[i].to);
                    }
                    pre[edge[i].to]=t;
                    fpre[edge[i].to]=i;
                }
            }
        }
        //printf("1
    ");
        if(dis[T]>=INF)return 0;
        return 1;
    }
    void MCMF(int S,int T)
    {
        int i;
        int ans=0;
        while(spfa(S,T))
        {
            int minflow=INF;
            for(i=T; i!=0; i=pre[i])
            {
                if(minflow>edge[fpre[i]].v)
                    minflow=edge[fpre[i]].v;
            }
            ans+=dis[T]*minflow;
            for(i=T; i!=0; i=pre[i])
            {
                edge[fpre[i]].v-=minflow;
                edge[edge[fpre[i]].flag].v+=minflow;
            }
        }
        printf("%d
    ",-ans);
    }
    int main()
    {
        int i,j;
        while(~scanf("%d%d",&n,&k))
        {
            N=n*n;
            index=1;
            memset(head,-1,sizeof(head));
            int cnt=0;
            for(i=0; i<n; i++)
                for(j=0; j<n; j++)
                {
                    map[i][j].cnt=++cnt;
                    scanf("%d",&map[i][j].v);
                }
            for(i=0; i<n; i++)
            {
                for(j=0; j<n; j++)
                {
                    add(map[i][j].cnt,map[i][j].cnt+N,1,-map[i][j].v);
                    if(ok(i,j,i,j+1))
                    {
                        add(map[i][j].cnt,map[i][j+1].cnt,INF,0);
                        add(map[i][j].cnt,map[i][j+1].cnt+N,INF,0);
                        add(map[i][j].cnt+N,map[i][j+1].cnt,INF,0);
                        add(map[i][j].cnt+N,map[i][j+1].cnt+N,INF,0);
                    }
                    if(ok(i,j,i+1,j))
                    {
                        add(map[i][j].cnt,map[i+1][j].cnt,INF,0);
                        add(map[i][j].cnt,map[i+1][j].cnt+N,INF,0);
                        add(map[i][j].cnt+N,map[i+1][j].cnt,INF,0);
                        add(map[i][j].cnt+N,map[i+1][j].cnt+N,INF,0);
                    }
                }
            }
            add(0,1,k,0);
            add(map[n-1][n-1].cnt,2*N,k,0);
            int S=0,T=n*n*2;
            MCMF(S,T);
        }
    }
  • 相关阅读:
    Android 代码规范大全
    python爬虫学习系列
    python接口测试(requests+excel)
    Jenkins系列
    postman系列
    利用Faker生成常用的测试数据
    python基础在实际写脚本过程中的一些简单运用(一)
    charles实践-通过断点调试修改request
    关于fiddler抓包的一些配置
    c#winform主题实现的一个方法
  • 原文地址:https://www.cnblogs.com/sweat123/p/4905466.html
Copyright © 2011-2022 走看看