题意就是

给一个50 * 50的矩阵

然后给出每行每列元素的和

和一个初始矩阵

矩阵中有些是未知,有些是已知

然后我们求目标矩阵就是把能确定的元素的值求出来,实在不能确定的就置为-1

所有矩阵元素的值在0-100之间

看到范围很小。

第一反应是求一个最大流

先把已经给出的元素都从每行每列的和中减掉。

然后左边为行结点,右边为列结点

然后源点向行结点连边

列结点向汇点连边

行和列中如果对应的元素未知就连一下,流向上限是100

然后这样我们就得到了一个可行解

但是可能有多解怎么办

对于一个可能多解的元素

如果我们将这个元素的值固定住。

然后建立一个超级源点与该行结点连边。

该列结点与超级汇点连边。

流量都是1,

跑一遍看看有没有增广路。

如果有,显然这个位置的值是可以改变的,就是多解,然后我们把这个位置的元素值-1,因为我们刚才增广了,其他有元素的值增加了1,所以

为了保持流量的平衡,这个位置的元素要减1

但是这样还不行。

我们想想。

如果该位置的值现在是0怎么办。

他没法减掉1。

所以我们就要想想残余网络了。

既然他没法减掉1,就让他想办法+1,让别的元素-1去

那么我们可以用一个超级源点连接列结点。

行结点连接超级汇点

跑最大流,看有没有增广路。

也就是看他的残余网络能不能减掉1.即它自身+1

如果有增广路,跟之前一样,更新一下边

#include <iostream>
#include <cstdio>
#include <cstring>
#include <string>
#include <algorithm>
#include <cstdlib>
#include <cmath>
#define MAXN 106
#define MAXM 211111
#define INF 1111111111
using namespace std;
struct EDGE
{
int v, next;
int w;
}edge[MAXM];
int head[MAXN], e;
void init()
{
memset(head, -1, sizeof(head));
e = 0;
}
void add(int u, int v, int w)
{
edge[e].v = v;
edge[e].w = w;
edge[e].next = head[u];
head[u] = e++;
edge[e].v = u;
edge[e].w = 0;
edge[e].next = head[v];
head[v] = e++;
}
int n;
int h[MAXN];
int gap[MAXN];
int src, des;
int tt[111][111];
int dfs(int pos, int cost)
{
if(pos == des) return cost;
int j, minh = n - 1;
int lv = cost, d;
for(j = head[pos]; j != -1; j = edge[j].next)
{
int v = edge[j].v;
int w = edge[j].w;
if(w > 0)
{
if(h[v] + 1 == h[pos])
{
if(lv < edge[j].w) d = lv;
else d = edge[j].w;
d = dfs(v, d);
edge[j].w -= d;
edge[j ^ 1].w += d;
lv -= d;
if(h[src] >= n) return cost - lv;
if(lv == 0) break;
}
if(h[v] < minh) minh = h[v];
}
}
if(lv == cost)
{
--gap[h[pos]];
if(gap[h[pos]] == 0) h[src] = n;
h[pos] = minh + 1;
++gap[h[pos]];
}
return cost - lv;
}
int sap()
{
int ret = 0;
memset(gap, 0, sizeof(gap));
memset(h, 0, sizeof(h));
gap[0] = n;
while(h[src] < n) ret += dfs(src, INF);
return ret;
}
int nt, m;
int col[55], row[55];
int val[55][55], id[55][55];
int ans[55][55];
int vis[55][55]; int main()
{
//freopen("C:/C.in", "r", stdin);
//freopen("C:/C2.out", "w", stdout);
while(scanf("%d%d", &nt, &m) != EOF)
{
if(!nt && !m) break;
for(int i = 1; i <= nt; i++)
for(int j = 1; j <= m; j++)
scanf("%d", &val[i][j]);
for(int i = 1; i <= nt; i++) scanf("%d", &row[i]);
for(int i = 1; i <= m; i++) scanf("%d", &col[i]);
memset(ans, -1, sizeof(ans));
for(int i = 1; i <= nt; i++)
for(int j = 1; j <= m; j++)
{
if(val[i][j] != -1)
{
row[i] -= val[i][j];
col[j] -= val[i][j];
ans[i][j] = val[i][j];
}
} init();
src = nt + m + 1;
des = nt + m + 2;
n = des;
int S = nt + m + 3;
int T = nt + m + 4; for(int i = 1; i <= nt; i++)
for(int j = 1; j <= m; j++)
{
if(ans[i][j] == -1)
{
id[i][j] = e;
add(i, j + nt, 100);
}
}
for(int i = 1; i <= nt; i++)
{
add(src, i, row[i]);
}
for(int j = 1; j <= m; j++)
{
add(j + nt, des, col[j]);
}
sap();
n = T;
memset(vis, 0, sizeof(vis));
for(int i = 1; i <= nt; i++)
for(int j = 1; j <= m; j++)
if(ans[i][j] != -1) vis[i][j] = 2;
src = S;
des = T;
for(int i = 1; i <= nt; i++)
{
for(int j = 1; j <= m; j++)
{
if(vis[i][j] == 2) continue;
int te = id[i][j];
int tmp = edge[te ^ 1].w;
edge[te].w = edge[te ^ 1].w = 0;
int flag = 1;
int le = e;
add(src, i, 1);
int me = e;
add(j + nt, des, 1);
if( tmp && sap()) flag = 0, tmp--;
edge[le].w = edge[le ^ 1].w = 0;
edge[me].w = edge[me ^ 1].w = 0;
le = e;
add(src, j + nt, 1);
me = e;
add(i, des, 1);
if(flag && 100 - tmp > 0&& sap()) flag = 0, tmp++;
edge[le].w = edge[le ^ 1].w = 0;
edge[me].w = edge[me ^ 1].w = 0; edge[te ^ 1].w = tmp;
edge[te].w = 100 - tmp;
vis[i][j] = flag;
} }
for(int i = 1; i <= nt; i++)
for(int j = 1; j <= m; j++)
{
if(vis[i][j] == 2)
{
if(ans[i][j] != -1) printf("%d", ans[i][j]);
}
else
{
if(vis[i][j]) printf("%d", edge[id[i][j] ^ 1].w);
else printf("-1");
}
if(j < m) printf(" ");
else printf("\n");
} }
return 0;
}

Uvalive 4865 Data Recovery 最大流的更多相关文章

  1. DRA(Data Recovery Advisor)的使用

    关于DRA的官方描述: The simplest way to diagnose and repair database problems is to use the Data Recovery Ad ...

  2. VMware Data Recovery备份恢复vmware虚拟机

    VMware Data Recovery 是VMware虚拟机备份工具,可创建虚拟机备份,同时不会中断虚拟机的使用或虚拟机提供的数据和服务.Data Recovery 管理现有备份,并在这些备份过时后 ...

  3. 转:主流数据恢复软件——EasyRecovery/Ashampoo Undeleter/Wise Data Recovery/Recuva/Undelete 360

    转自:Baidu 空间 2012-10-05 13:57 主流数据恢复软件——EasyRecovery/Ashampoo Undeleter/Wise Data Recovery/Recuva/Und ...

  4. Data Recovery Advisor (DRA)

    数据恢复指导Data Recovery Advisor (DRA)的适用场景:Data Recovery Advisor 是11g 新特性,是Oracle 顾问程序架构的一部分,它会在遇到错误时自动收 ...

  5. Data Recovery Advisor(数据恢复顾问)

    Data Recovery Advisor 是11g新特性,是Oracle顾问程序架构的一部分,它会在遇到错误时自动收集有关故障信息.如果主动运行Data Recovery Advisor,通常可以在 ...

  6. Percona Data Recovery Tool for InnoDB工具恢复单表的案例

    今天上班有个朋友询问我,相关Percona Data Recovery Tool for InnoDB恢复数据中的一些问题,比如说delete,没法恢复数据,原先做过类似的异常处理就,再次模拟了下相关 ...

  7. 使用Percona Data Recovery Tool for InnoDB恢复数据

      运维工作中难免会发生一些误操作,当数据库表被误操作删除需要紧急恢复,或者没有备份时,Percona Data Recovery Tool for InnoDB这个工具也已提供一些便捷的恢复. 当然 ...

  8. Android Data Recovery for Mac(安卓数据恢复软件)破解版安装

    1.软件简介    Android Data Recovery 是 macOS 系统上一款 Android 设备数据恢复软件,能够帮助我们在 mac 电脑上对 Android 设备进行数据恢复,文档. ...

  9. Aiseesoft Data Recovery 1.1.6 专业数据恢复软件破解版

    Aiseesoft Data Recovery是专业的数据恢复软件,它可以帮助你恢复几乎所有删除/丢失的文件,如照片,文件,电子邮件,音频,视频且支持从计算机,硬盘驱动器,闪存驱动器,存储卡,数码相机 ...

随机推荐

  1. Java-利用spring发送邮件

    最近项目中需要发送邮件的功能,于是百度一大把例子.但是有很多都是一样的,一点特点都没有.所以决定整理一番.         在spring2.X以后的版本就提供了org.springframework ...

  2. OpenERP 安装在Windows server上时间显示不对的解决办法

    这个问题一直困扰着我,后台设置关于时区也参考过一些文章(如改配置文件),显示时间总是差8个小时,后来看了上面的文章后才搞定,虽然那个是解决OE7.0的,但原理一样: server\openerp\ad ...

  3. Core Text

    Core Text 本文所涉及的代码你可以在这里下载到 https://github.com/kejinlu/CTTest,包含两个项目,一个Mac的NSTextView的测试项目,一个iOS的Cor ...

  4. 编译安装apache+php(加常见问题解决)

    [编译apache]./configure --prefix=/usr/local/lamp/httpd -with-apr=/usr/local/apr -with-apr-util=/usr/lo ...

  5. CocoStudio UI 编辑器的使用

    详细教程:http://www.cocoachina.com/bbs/read.php?tid=161567 Table of Contents 1 游戏中的 UI 1.1 基于 Cocos2d-x ...

  6. js COOKIE 记住帐号或者uuid

    当开始接到这个任务的时候,我对cookie还是没多少了解的,而uuid的生成也是一无所知.但是当你发现这个网址http://stackoverflow.com/questions/105034/how ...

  7. [转]vi与vim的区别

    一直用着vi,有朋友劝我用vim,那么它们有什么区别呢? 简单点来说,它们都是多模式编辑器, 不同的是vim 是vi的升级版本,它不仅兼容vi的所有指令, 而且还有一些新的特性在里面. vim的这些优 ...

  8. vim的保存误认为utf8问题

     用vim改脚本改到一处写到'太原':w一下,再打开,,结果给乱码了...我默认sql是用cp936的,,,想到到和记录本的联通问题一样....   可能会问我为什么不用utf8,,,,因为ms200 ...

  9. ubuntu 上更新安装 openoffice.org3的过程

    方法一:手动安装1首先在“应用成程序--添加/删除”里卸载openoffice 2.4,可能openoffice.org 2.4 Draw因为关联而无法卸载,忽略,卸载其他几项.然后在中文官方网上下载 ...

  10. 5、NFC概述

    什么是NFC NFC(Near Field Communication,近场通信),是一种数据传输技术.但与Wi-Fi.蓝牙.红外线等数据传输技术的一个主要差异就是有效距离一般不能超过4厘米. NFC ...