首页 > Windows开发 > 详细

HDU 4888 Redraw Beautiful Drawings 网络流 建图

时间:2014-07-29 22:03:03      阅读:447      评论:0      收藏:0      [点我收藏+]

题意:

给定n, m, k

下面n个整数 a[n]

下面m个整数 b[n]

用数字[0,k]构造一个n*m的矩阵

若有唯一解则输出这个矩阵,若有多解输出Not Unique,若无解输出Impossible


思路:网络流,,,

n行当成n个点,m列当成m个点

从行-列连一条流量为k的边,然后源点-行连一条a[i]的边, 列-汇点 流量为b[i]


瞎了,该退役了 T^T


#include<stdio.h>
#include<string.h>
#include<iostream>
#include<math.h>
#include<algorithm>
#include<queue>
#include<vector>
using namespace std;

#define ll int

#define N 1005
#define M 200000
#define inf 107374182
#define inf64 1152921504606846976
struct Edge{
    ll from, to, cap, nex;
}edge[M*2];//注意这个一定要够大 不然会re 还有反向弧

ll head[N], edgenum;
void add(ll u, ll v, ll cap, ll rw = 0){ //如果是有向边则:add(u,v,cap); 如果是无向边则:add(u,v,cap,cap);
    Edge E = { u, v, cap, head[u]};
    edge[ edgenum ] = E;
    head[u] = edgenum ++;

    Edge E2= { v, u, rw,  head[v]};
    edge[ edgenum ] = E2;
    head[v] = edgenum ++;
}
ll sign[N];
bool BFS(ll from, ll to){
    memset(sign, -1, sizeof(sign));
    sign[from] = 0;

    queue<ll>q;
    q.push(from);
    while( !q.empty() ){
        ll u = q.front(); q.pop();
        for(ll i = head[u]; i!=-1; i = edge[i].nex)
        {
            ll v = edge[i].to;
            if(sign[v]==-1 && edge[i].cap)
            {
                sign[v] = sign[u] + 1, q.push(v);
                if(sign[to] != -1)return true;
            }
        }
    }
    return false;
}
ll Stack[N], top, cur[N];
ll Dinic(ll from, ll to){
    ll ans = 0;
    while( BFS(from, to) )
    {
        memcpy(cur, head, sizeof(head));
        ll u = from;      top = 0;
        while(1)
        {
            if(u == to)
            {
                ll flow = inf, loc;//loc 表示 Stack 中 cap 最小的边
                for(ll i = 0; i < top; i++)
                    if(flow > edge[ Stack[i] ].cap)
                    {
                        flow = edge[Stack[i]].cap;
                        loc = i;
                    }

                    for(ll i = 0; i < top; i++)
                    {
                        edge[ Stack[i] ].cap -= flow;
                        edge[Stack[i]^1].cap += flow;
                    }
                    ans += flow;
                    top = loc;
                    u = edge[Stack[top]].from;
            }
            for(ll i = cur[u]; i!=-1; cur[u] = i = edge[i].nex)//cur[u] 表示u所在能增广的边的下标
                if(edge[i].cap && (sign[u] + 1 == sign[ edge[i].to ]))break;
            if(cur[u] != -1)
            {
                Stack[top++] = cur[u];
                u = edge[ cur[u] ].to;
            }
            else
            {
                if( top == 0 )break;
                sign[u] = -1;
                u = edge[ Stack[--top] ].from;
            }
        }
    }
    return ans;
}
void init(){memset(head,-1,sizeof head);edgenum = 0;}


int n, m, k;
int a[500], b[500], suma, sumb;
int mp[505][505], dou[505][505]; //dou[0][i] i这列存在一个可增的点
int hehe(){
    if(suma != sumb)return -1;
    init();
    int from = 0, to = n+m + 10;
    for(int i = 1; i <= n; i++)
        for(int j = 1; j <= m; j++)
            add(i, n+j, k);
    for(int i = 1; i <= n; i++)
        add(from, i, a[i]);
    for(int i = 1; i <= m; i++)
        add(n+i, to, b[i]);
    int flow = Dinic(from, to);
    if(flow != suma) return -1;
    int tt = 1;
    for(int i = 1; i <= n; i++)for(int j = 1; j <= m; j++, tt+=2)
        mp[i][j] = edge[tt].cap;
    memset(dou, 0, sizeof dou);
    for(int i = 1; i <= n; i++)
    {
        for(int j = 1; j <= m; j++)
            for(int z = j+1; z <= m; z++)
            {
                bool v1=0,v2=0;
                if(mp[i][j]!=k&&mp[i][z]!=0)
                {
                    if(dou[z][j])return 0;
                    v1=1;
                }
                if(mp[i][j]!=0&&mp[i][z]!=k)
                {
                    if(dou[j][z])return 0;
                    v2=1;
                }
                if(v1)dou[j][z]=1;
                if(v2)dou[z][j]=1;
            }

    }
    return 1;
}
void input(){
    suma = sumb = 0;
    for(int i = 1; i <= n; i++)scanf("%d",&a[i]), suma += a[i];
    for(int i = 1; i <= m; i++)scanf("%d",&b[i]), sumb += b[i];
}
int main(){
    int u, v, i, j;
    while(~scanf("%d %d %d",&n,&m,&k)) {
        input();
        int ans = hehe();
        if(ans == -1)puts("Impossible");
        else if(ans == 0)puts("Not Unique");
        else
        {
            puts("Unique");
            for(i = 1; i <= n; i++)
                for(j = 1; j <= m; j++)
                    printf("%d%c",mp[i][j],j==m?'\n':' ');
        }
    }
    return 0;
}
/*
2 3 8
13 16
3 11 15

2 4 8
15 16
3 11 15 2

3 4 8
15 16 10
4 12 18 6

3 4 8
15 16 10
4 13 18 6

3 5 8
16 16 11
4 13 18 6 2

3 4 1
1 3 4
3 2 1 2

*/


HDU 4888 Redraw Beautiful Drawings 网络流 建图,布布扣,bubuko.com

HDU 4888 Redraw Beautiful Drawings 网络流 建图

原文:http://blog.csdn.net/qq574857122/article/details/38276447

(0)
(0)
   
举报
评论 一句话评论(0
关于我们 - 联系我们 - 留言反馈 - 联系我们:wmxa8@hotmail.com
© 2014 bubuko.com 版权所有
打开技术之扣,分享程序人生!