链接:

http://poj.org/problem?id=3436

http://acm.hust.edu.cn/vjudge/contest/view.action?cid=82835#problem/A

题目大意:

一个生产电脑的工厂,有多个机器生产一台电脑,大家都知道一台电脑有多个零部件,这些机器在一个流水线,有些机器工作的前提是需要一些零部件的存在才能继续工作生产,每个机器都有输入设备和输出设备,输入设备意思是根据这几个零部件的存在状态才能使此机器工作,状态包含(0,该零部件不需要存在,1,该零部件必须存在,2,该零部件可有可无)。输出设备意思为该机器工作之后生产的电脑的零部件状态情况(0,没有生产1,生成出该零件)。p,代表每个电脑零部件的个数,m代表几个机器,还给出了每个机器最大的限度能够同时处理机器的个数。根据输入算出最多可以生产的电脑数量和路径个数和路径的详情。

样例说明:
3 4
1号: 15    0 0 0 --> 0 1 0
2号: 10    0 0 0 --> 0 1 1
3号: 30    0 1 2 --> 1 1 1
4号: 3      0 2 1 --> 1 1 1

注意:因为每个生产线的生产能力有限,所以需要拆点,防止超出他的生产能力,比如下图如果不拆点结果就会使20,实际上是10


借鉴别人的代码:

#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
#include<queue>
using namespace std; #define N 110
#define INF 0x3fffffff int g[N][N], flow[N], p[N], Line[N][4];
int In[N][20];///输入信息
int backup[N][N]; ///备份图
int nn; ///点的个数,编号0-nn包括源点和汇点 queue<int>q; int BFS(int Start, int End)
{
int i, t;
while(q.size()) q.pop(); memset(p, -1, sizeof(p));
p[Start] = 0;
flow[Start] = INF;///源点可以有无穷的流流进
q.push(Start); while(q.size())
{
t = q.front();
q.pop();
if(t==End) break;
///枚举所有的点,如果点的编号起始点有变化可以改这里
for(i=0; i<=nn; i++)
{
if(i!=Start && p[i]==-1 && g[t][i])
{
flow[i] = flow[t] < g[t][i]?flow[t]:g[t][i];
q.push(i);
p[i]=t;
}
}
}
if(p[End]==-1) return -1; ///找不到汇点上去了,找不到增广路径了
return flow[End];
}
int Edmonds_Karp(int Start, int End)
{
int MaxFlow=0;
int step, now, pre; while((step=BFS(Start, End))!=-1)
{
MaxFlow += step;
now = End;
while(now!=Start)
{
pre = p[now];
g[pre][now] -= step;
g[now][pre] += step;
now = pre;
}
}
return MaxFlow;
} int main()
{
int p, n, Start, End; while(scanf("%d%d", &p, &n)!=EOF)
{
memset(g, 0, sizeof(g));
for(int i=1; i<=n; i++)
{
for(int j=0; j<2*p+1; j++)
scanf("%d", &In[i][j]);
}
for(int i=1; i<=n; i++)
g[2*i-1][2*i] = In[i][0]; nn = 2*n + 1;
Start = 0; ///源点
End = nn; ///汇点 for(int i=1; i<=n; i++)
{
bool flag_s = true;
bool flag_t = true; for(int j=0; j<p; j++)
{
if(In[i][j+1]==1) flag_s = false;
if(In[i][j+1+p]==0) flag_t = false;
}
if(flag_s) g[0][2*i-1] = INF;
if(flag_t) g[2*i][nn] = INF; for(int j=1; j<=n; j++)
if(i!=j)
{
bool flag = true;
for(int k=0; k<p; k++)
if((In[i][k+p+1]==0 && In[j][k+1]==1) || (In[i][k+p+1]==1 && In[j][k+1]==0))
{
flag = false;
break;
}
if(flag) g[2*i][2*j-1] = min(In[i][0], In[j][0]);
}
} memcpy(backup, g, sizeof(g)); ///先把图备份下来
printf("%d ", Edmonds_Karp(Start, End)); int tol=0;
for(int i=1; i<=n; i++)
for(int j=1; j<=n; j++)
{
if(g[2*i][2*j-1]<backup[2*i][2*j-1])
{
Line[tol][0]=i;
Line[tol][1]=j;
Line[tol++][2]= backup[2*i][2*j-1] - g[2*i][2*j-1];
}
} printf("%d\n", tol);
for(int i=0; i<tol; i++)
printf("%d %d %d\n", Line[i][0], Line[i][1], Line[i][2]);
}
return 0;
}

代码:

#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
#include<queue>
using namespace std; #define N 110
#define INF 0x3fffffff struct node
{
int Flow, In[N], Out[N];
}a[N]; int s[N][N], s1[N][N], Layer[N];
int n, p; /// 重点在建图和拆点, 别的似乎没太多的不同 void Init()
{
memset(s, , sizeof(s));
memset(s1, , sizeof(s1));
memset(Layer, -, sizeof(Layer)); for(int i=; i<=p; i++)///要把0作为源点, 把n+2作为汇点
{
a[].In[i] = ;
a[].Out[i] = ;
a[n+].In[i] = ;
a[n+].Out[i] = ;
}
a[].Flow = INF;
a[n+].Flow = INF;
} bool CanLink(node n1, node n2)
{
for(int i=; i<=p; i++)
{
if(n1.Out[i]!=n2.In[i] && n2.In[i]!=)
return false;
}
return true;
}
bool BFS(int Start, int End)
{
int used[N]={};
used[Start] = true, Layer[Start] = ;
queue<int>Q; Q.push(Start); while(Q.size())
{
int u = Q.front(); Q.pop(); if(u==End) return true; for(int i=; i<=End; i++)
{
if(s[u][i] && !used[i])
{
used[i] = true;
Layer[i] = Layer[u] + ;
Q.push(i);
}
}
}
return false;
}
int DFS(int u, int MaxFlow, int End)
{
if(u==End) return MaxFlow; int uflow = ; for(int i=; i<=End; i++)
{
if(Layer[u]+ == Layer[i] && s[u][i])
{
int flow = min(s[u][i], MaxFlow-uflow);
flow = DFS(i, flow, End); s[u][i] -= flow;
s[i][u] += flow;
uflow += flow;
} if(uflow == MaxFlow)
break;
} if(uflow==)
Layer[u] = -; return uflow;
}
int Dinic(int Start, int End)
{
int MaxFlow = ; while(BFS(Start, End)==true)
MaxFlow += DFS(Start, INF, End); return MaxFlow;
} int main()
{
while(scanf("%d%d", &p, &n)!=EOF)
{
int i, j; Init();
for(i=; i<=n+; i++)
{
scanf("%d", &a[i].Flow);
for(j=; j<=p; j++)
scanf("%d", &a[i].In[j]);
for(j=; j<=p; j++)
scanf("%d", &a[i].Out[j]);
} n+=; for(i=; i<=n; i++) /// 拆点并保存原来的路径值, 每个点都要拆分成两个点
for(j=; j<=n; j++) ///好神奇呀,以前都没遇到过
{
if(i==j) ///s1[i+n][j] = s[i+n][j] = a[i].Flow;
s1[j][i+n] = s[j][i+n] = a[i].Flow;
else if(i!=j && CanLink(a[i], a[j])==true)
s1[i+n][j] = s[i+n][j] = a[i].Flow;
} int MaxFlow = Dinic(, n*);
int k=, x[N], y[N], flow[N]; for(i=; i<n; i++)
for(j=; j<n; j++)
{
if(s[i+n][j]<s1[i+n][j])
{
x[k] = i;
y[k] = j;
flow[k++] = s1[i+n][j] - s[i+n][j];
}
} printf("%d %d\n", MaxFlow, k); for(i=; i<k; i++)
printf("%d %d %d\n", x[i]-, y[i]-, flow[i]); }
return ;
}

(网络流)ACM Computer Factory --POJ --3436的更多相关文章

  1. A - ACM Computer Factory POJ - 3436 网络流

    A - ACM Computer Factory POJ - 3436 As you know, all the computers used for ACM contests must be ide ...

  2. ACM Computer Factory - poj 3436 (最大流)

      Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 5949   Accepted: 2053   Special Judge ...

  3. ACM Computer Factory POJ - 3436 网络流拆点+路径还原

    http://poj.org/problem?id=3436 每台电脑有$p$个组成部分,有$n$个工厂加工电脑. 每个工厂对于进入工厂的半成品的每个组成部分都有要求,由$p$个数字描述,0代表这个部 ...

  4. A - ACM Computer Factory - poj 3436(最大流)

    题意:有一个ACM工厂会生产一些电脑,在这个工厂里面有一些生产线,分别生产不同的零件,不过他们生产的电脑可能是一体机,所以只能一些零件加工后别的生产线才可以继续加工,比如产品A在生产线1号加工后继续前 ...

  5. POJ 3436 ACM Computer Factory (网络流,最大流)

    POJ 3436 ACM Computer Factory (网络流,最大流) Description As you know, all the computers used for ACM cont ...

  6. POJ 3436:ACM Computer Factory 网络流

    ACM Computer Factory Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 6247   Accepted: 2 ...

  7. Poj 3436 ACM Computer Factory (最大流)

    题目链接: Poj 3436 ACM Computer Factory 题目描述: n个工厂,每个工厂能把电脑s态转化为d态,每个电脑有p个部件,问整个工厂系统在每个小时内最多能加工多少台电脑? 解题 ...

  8. POJ 3464 ACM Computer Factory

    ACM Computer Factory Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 4829 Accepted: 1641 ...

  9. POJ-3436 ACM Computer Factory(网络流EK)

    As you know, all the computers used for ACM contests must be identical, so the participants compete ...

随机推荐

  1. JAVA中会存在内存泄露吗

    所谓内存泄露就是指一个不再被程序使用的对象或变量一直被占据在内存中.java中有垃圾回收机制,它可以保证一对象不再被引用的时候,即对象编程了孤儿的时候,对象将自动被垃圾回收器从内存中清除掉.由于Jav ...

  2. linux下面得小数计算

    可以通过命令行向awk中传递参数. 这样子传递进去的参数,在awk命令中可以访问.每一项都必须作为单一的一个参数来解释.所以,等号之间不能有空格. 比如说我们传递进去一个学生名字,想查这个学生的得分. ...

  3. spring-boot PageHelper

    分页插件PageHelper 参看了pagehelper-spring-boot,使用起来非常放方便,关于更多PageHelper可以点击https://github.com/pagehelper/M ...

  4. Nexus 使用配置

    Nexus使用的一些基本设置 1.更改中央仓库地址为私服地址 既然我们配置了私服,那么相应的,我们的项目就应该使用Nexus的地址(Public Repository)来下载jar包 1.1.基于PO ...

  5. xpath的层级与逻辑定位

    xpath的层级与逻辑定位: 之前我们是通过class和id,name,如果我们所需要的元素没有class,id,name这样的元素,怎么定位呢 1.在不使用xpath情况下:元素没有属性值得时候怎么 ...

  6. mysql的外键知识

    外键的作用 1.用来约束两张表中的字段 2.外键也可以用来实现一对多 我们先举一个这样的例子,让大家对外键有一个基本的认识 当前我们有一个需求就是,需要创建一张表,这张表要包括“姓名”,“年龄”,“工 ...

  7. docker搭建nginx

    在Docker下部署Nginx,包括: 部署一个最简单的Nginx,可以通过端口访问默认的网站 设置记录访问和错误日志的路径 以交互模式创建centos容器,如果本地没有镜像,会从仓库获取, 等待即可 ...

  8. win下安装composer

    1.在官网上下载composer的安装程序. https://getcomposer.org/ 2.双击运行Composer-Setup.exe程序,选择你自已的php安装目录打到php.exe.   ...

  9. go实现的简易TCP的客户端和服务器

    今天介绍golang版本的通信基础:基于TCP的客户端和服务器实现,参考书籍:The Way To Go 那时学习java的时候也是做过通信的,当时是socket编程,服务器监听某一个端口,然后客户机 ...

  10. PAT 1020 月饼 (25)(精简版代码+思路+推荐测试用例)

    1020 月饼 (25)(25 分)提问 月饼是中国人在中秋佳节时吃的一种传统食品,不同地区有许多不同风味的月饼.现给定所有种类月饼的库存量.总售价.以及市场的最大需求量,请你计算可以获得的最大收益是 ...