博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
hdu4975 网络流解方程组(网络流+dfs判环或矩阵DP)
阅读量:5111 次
发布时间:2019-06-13

本文共 5655 字,大约阅读时间需要 18 分钟。

A simple Gaussian elimination problem.

Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others)
Total Submission(s): 579    Accepted Submission(s): 194
Problem Description
Dragon is studying math. One day, he drew a table with several rows and columns, randomly wrote numbers on each elements of the table. Then he counted the sum of each row and column. Since he thought the map will be useless after he got the sums, he destroyed the table after that.
However Dragon's mom came back and found what he had done. She would give dragon a feast if Dragon could reconstruct the table, otherwise keep Dragon hungry. Dragon is so young and so simple so that the original numbers in the table are one-digit number (e.g. 0-9).
Could you help Dragon to do that?
 
Input
The first line of input contains only one integer, T(<=30), the number of test cases. Following T blocks, each block describes one test case.
There are three lines for each block. The first line contains two integers N(<=500) and M(<=500), showing the number of rows and columns.
The second line contains N integer show the sum of each row.
The third line contains M integer show the sum of each column.
 
Output
Each output should occupy one line. Each line should start with "Case #i: ", with i implying the case number. For each case, if we cannot get the original table, just output: "So naive!", else if we can reconstruct the table by more than one ways, you should output one line contains only: "So young!", otherwise (only one way to reconstruct the table) you should output: "So simple!".
 
Sample Input
 
3 1 1 5 5 2 2 0 10 0 10 2 2 2 2 2 2
 
Sample Output
 
Case #1: So simple! Case #2: So naive! Case #3: So young!

题目意思很简单:就是给出一个矩阵的行和和列和,矩阵中的每个元素都是0-9,问原矩阵是否存在,是否唯一;

分析:网络流求解,如果最大流=所有元素的和则有解;利用残留网络判断是否唯一,方法有两种,第一种是深搜看看是否存在正边权的环,至少3个点构成的环,第二种是用矩阵dp,假如某行的i列元素<9,j列元素>0,而另一行的i列元素>0,j列元素<9,那么答案不是唯一的,因为主对角线的 两个元素可以增大1,而副对角线的两个元素可以减小1,可以明显看出有多个答案;

比赛时的程序:

#include"stdio.h"#include"string.h"#include"iostream"#include"map"#include"string"#include"queue"#include"stdlib.h"#include"math.h"#define M 1900#define eps 1e-10#define inf 100000000using namespace std;struct node{    int u,v,w,next;}edge[600000];int t,head[M],row[M],col[M],q[M],dis[M],work[M],use[M];void init(){    t=0;    memset(head,-1,sizeof(head));}void add(int u,int v,int w){    edge[t].u=u;    edge[t].v=v;    edge[t].w=w;    edge[t].next=head[u];    head[u]=t++;    edge[t].u=v;    edge[t].v=u;    edge[t].w=0;    edge[t].next=head[v];    head[v]=t++;}int bfs(int S,int T){    int rear=0;    memset(dis,-1,sizeof(dis));    dis[S]=0;    q[rear++]=S;    for(int i=0;i
之后用矩阵dp做的:

#include"stdio.h"#include"string.h"#include"iostream"#include"map"#include"string"#include"queue"#include"stdlib.h"#include"math.h"#define M 1900#define eps 1e-10#define inf 1000000000#define mod 2333333using namespace std;struct node{    int u,v,w,next;}edge[600000];int t,head[M],work[M],use[M],dis[M],mp[555][555],G[555][555],row[555],col[555];void init(){    t=0;    memset(head,-1,sizeof(head));}void add(int u,int v,int w){    edge[t].u=u;    edge[t].v=v;    edge[t].w=w;    edge[t].next=head[u];    head[u]=t++;    edge[t].u=v;    edge[t].v=u;    edge[t].w=0;    edge[t].next=head[v];    head[v]=t++;}int bfs(int S,int T){    memset(dis,-1,sizeof(dis));    queue
q; dis[S]=0; q.push(S); while(!q.empty()) { int u=q.front(); q.pop(); for(int i=head[u];i!=-1;i=edge[i].next) { int v=edge[i].v; if(edge[i].w&&dis[v]==-1) { dis[v]=dis[u]+1; q.push(v); if(v==T) return 1; } } } return 0;}int dfs(int cur,int a,int T){ if(cur==T)return a; for(int &i=work[cur];i!=-1;i=edge[i].next) { int v=edge[i].v; if(edge[i].w&&dis[v]==dis[cur]+1) { int tt=dfs(v,min(edge[i].w,a),T); if(tt) { edge[i].w-=tt; edge[i^1].w+=tt; return tt; } } } return 0;}int Dinic(int S,int T){ int ans=0; while(bfs(S,T)) { memcpy(work,head,sizeof(head)); while(int tt=dfs(S,inf,T)) ans+=tt; } return ans;}int judge(int n,int m){ int k=0,i,j; for(i=1;i<=n;i++) { for(j=1;j<=m;j++) { G[i][j]=edge[k^1].w; k+=2; } } memset(mp,0,sizeof(mp)); for(i=1;i<=n;i++) { if(row[i]==0||row[i]==9*m)continue; for(j=1;j<=m;j++) { if(col[j]==0||col[j]==9*n)continue; for(k=j+1;k<=m;k++) { int f1=0,f2=0; if(G[i][j]<9&&G[i][k]>0) { if(mp[k][j]) return 1; f1++; } if(G[i][j]>0&&G[i][k]<9) { if(mp[j][k]) return 1; f2++; } if(f1)mp[j][k]=1; if(f2)mp[k][j]=1; } } } return 0;}int main(){ int T,m,n,kk=1,i,j; scanf("%d",&T); while(T--) { scanf("%d%d",&n,&m); int r=0; for(i=1;i<=n;i++) { scanf("%d",&row[i]); r+=row[i]; } int c=0; for(j=1;j<=m;j++) { scanf("%d",&col[j]); c+=col[j]; } printf("Case #%d: ",kk++); if(c!=r) { printf("So naive!\n"); continue; } int flag=0; for(i=1;i<=n;i++) if(9*m

转载于:https://www.cnblogs.com/mypsq/p/4348173.html

你可能感兴趣的文章
uiautomator_python使用汇总
查看>>
tomcat cluster session同步时保存map数据遇到的问题
查看>>
Javascript备忘录-枚举一个对象的所有属
查看>>
Asp.net MVC DefaultModelBinder分析
查看>>
KVM安装
查看>>
w3cschool -css
查看>>
《Entity Framework 6 Recipes》中文翻译系列 (10) -----第二章 实体数据建模基础之两实体间Is-a和Has-a关系建模、嵌入值映射 (转)...
查看>>
又是毕业季I
查看>>
涛涛的Party
查看>>
SQL Server 触发器
查看>>
Silverlight 5 系列学习之一
查看>>
最值栈
查看>>
EXTJS中文乱码
查看>>
POJ2226 Muddy Fields 二分匹配 最小顶点覆盖 好题
查看>>
POJ 2528 Mayor's posters 线段树+离散化
查看>>
将DataSet(DataTable)转换成JSON格式(生成JS文件存储)
查看>>
javascript日常学习小记
查看>>
Objective-C 学习笔记(Day 2)
查看>>
如何使用PHP显示在线Word文档
查看>>
Discuz常见小问题-如何设置163邮箱注册验证
查看>>