【网络流#8】POJ 3469 Dual Core CPU 最小割【ISAP模板】 - 《挑战程序设计竞赛》例题
【题意】有n个程序,分别在两个内核中运行,程序i在内核A上运行代价为ai,在内核B上运行的代价为bi,现在有程序间数据交换,如果两个程序在同一核上运行,则不产生额外代价,在不同核上运行则产生Cij的额外代价,问如何划分使得代价最小。
用最小的费用将对象划分为两个集合的问题,常常可以转换为最小割后顺利解决
建立源点与汇点,每个程序视为一个点,源点与在各个程序连一条边,最大流量为bi,汇点与各个程序连一条边,最大流量ai,对于有额外代价的程序,连一条双向边,流量为cij。
一开始用Dinic算法做,结果TLE,在POJ的discuss里看到了这篇http://www.cnblogs.com/zhsl/archive/2012/12/03/2800092.html
ISAP的效率要优于DINIC所以这道题需要使用ISAP。
ISAP的模板的话找到了这篇http://kenby.iteye.com/blog/945454相当不错所以借鉴了过来。
#include<cstdio>
#include<cstring>
#include<cmath>
#include<iostream>
#include<algorithm>
#include<set>
#include<map>
#include<stack>
#include<vector>
#include<queue>
#include<string>
#include<sstream>
#define eps 1e-9
#define ALL(x) x.begin(),x.end()
#define INS(x) inserter(x,x.begin())
#define FOR(i,j,k) for(int i=j;i<=k;i++)
#define MAXN 20010
#define MAXM 1800000
#define INF 0x3fffffff
using namespace std;
typedef long long LL;
int i,j,k,n,m,x,y,T,ans,big,cas,w;
bool flag; struct Edge{
int u,v,weight;
int next;
}edge[MAXM];
int head[MAXN]; /* head[u]表示顶点u第一条邻接边的序号, 若head[u] = -1, u没有邻接边 */
int current; /* 当前有多少条边 */ void add_edge(int u, int v, int weight)
{
/* 添加正向边u->v */
edge[current].u = u;
edge[current].v = v;
edge[current].weight = weight;
edge[current].next = head[u];
head[u] = current++;
/* 添加反向边v->u */
edge[current].u = v;
edge[current].v = u;
edge[current].weight = 0;
edge[current].next = head[v];
head[v] = current++;
} int isap(int s, int e)
{
int i,u,v,max_flow,aug,min_lev; /* 寻找增广路径的过程中, curedge[u]保存的是对于顶点u当前遍历的边, 寻找顶点u的邻接边时不用每次
* 都从head[u]开始找, 而是从curedge[u]开始找, 这样就减少了搜索次数
* 当增广路径找到后
* curedge保存的就是一条增广路径了, 比如
* 0---四-->1---六-->2--七--->3---八--->4 0,1,2,3,4是顶点号, 四六七八是边的序号
* curedge[0] = 四, curedge[1] = 六, ... curedge[3] = 8, curedge[i]即保存找过的轨迹
*/
int curedge[MAXN],parent[MAXN],level[MAXN]; /* count[l]表示对于属于层次l的顶点个数, 如果某个层次没有顶点了,
* 就出现断层, 意味着没有增广路径了, 这就是gap优化, 可以提前结束寻找过程
* augment[v]表示从源点到顶点v中允许的最大流量, 即这条路线的最小权重
*/
int count[MAXN],augment[MAXN]; memset(level,0,sizeof(level));
memset(count,0,sizeof(count));
//初始时每个顶点都从第一条边开始找
for (i=0;i<=n;i++)
{
curedge[i] = head[i];
}
max_flow=0;
augment[s]=INF;
parent[s]=-1;
u=s; while (level[s]<n) /* 不能写成level[s] < MAX_INT */
{
if (u==e) /* 找到一条增广路径 */
{
max_flow+=augment[e];
aug=augment[e];
//debug("找到一条增广路径, augment = %d\n", aug);
//debug("%d", e);
for (v=parent[e];v!=-1;v=parent[v]) /* 从后往前遍历路径 */
{
i=curedge[v];
//debug("<--%d", v);
edge[i].weight-=aug;
edge[i^1].weight+=aug; /* 如果i是偶数, i^1 = i+1, 如果i是奇数, i^1 = i-1 */
augment[edge[i].v]-=aug;
if (edge[i].weight==0) u=v; /* u指向增广后最后可达的顶点, 下次就从它继续找 */
}
//debug("\n");
}
/* 从顶点u往下找邻接点 */
for (i=curedge[u];i!=-1;i=edge[i].next) /* 从curedge[u]开始找, 而不是head[u]从头开始, curedge[u]保存的是上次找过的边 */
{
v=edge[i].v;
if (edge[i].weight>0 && level[u]==(level[v]+1)) /* 找到一条边就停止 */
{
augment[v]=min(augment[u],edge[i].weight);
curedge[u]=i;
parent[v]=u;
u=v;
break;
}
}
if (i==-1) /* 没有邻接点, 回溯到上一个点 */
{
if (--count[level[u]]==0)
{
//debug("顶点%d在level %d断层\n", u, level[u]);//GAP优化
break;
}
curedge[u]=head[u]; /* 顶点u的所有边都试过了,没有出路, 更新了u的level后, 又从第一条边开始找 */
//找出level最小的邻接点
min_lev=n;
for (i=head[u];i!=-1;i=edge[i].next)
{
if (edge[i].weight>0)
{
min_lev=min(level[edge[i].v],min_lev);
}
}
level[u]=min_lev+1;
count[level[u]]++;
//debug("顶点%d的level= %d\n", u, level[u]);
//debug("顶点%d走不通, 回到%d\n", u, edge[curedge[u]].u);
if (u!=s) u=parent[u]; /* 回退到上一个顶点 */
}
}
return max_flow;
} int main()
{
int m,u,v,w,a,b;
while (scanf("%d %d",&n,&m)!=EOF)
{
memset(edge,0,sizeof(edge));
memset(head,-1,sizeof(head));
current=0;
for (u=1;u<=n;u++)
{
scanf("%d %d",&a,&b);
add_edge(0,u,a);
add_edge(u,n+1,b);
}
while (m--)
{
scanf("%d %d %d",&u,&v,&w);
/* 如果调用函数添加边, 速度明显边慢 */
//add_edge(u, v, w);
//add_edge(v, u, w); /* 添加正向边u->v */
edge[current].u = u;
edge[current].v = v;
edge[current].weight = w;
edge[current].next = head[u];
head[u] = current++;
/* 添加反向边v->u */
edge[current].u = v;
edge[current].v = u;
edge[current].weight = w;
edge[current].next = head[v];
head[v] = current++;
}
n+=2;
printf("%d\n",isap(0,n-1));
}
return 0;
}
【网络流#8】POJ 3469 Dual Core CPU 最小割【ISAP模板】 - 《挑战程序设计竞赛》例题的更多相关文章
- poj 3469 Dual Core CPU——最小割
题目:http://poj.org/problem?id=3469 最小割裸题. 那个限制就是在 i.j 之间连双向边. 根据本题能引出网络流中二元关系的种种. 别忘了写 if ( x==n+1 ) ...
- POJ 3469 Dual Core CPU (最小割建模)
题意 现在有n个任务,两个机器A和B,每个任务要么在A上完成,要么在B上完成,而且知道每个任务在A和B机器上完成所需要的费用.然后再给m行,每行 a,b,w三个数字.表示如果a任务和b任务不在同一个机 ...
- poj 3469 Dual Core CPU 最小割
题目链接 好裸的题....... 两个cpu分别作为源点和汇点, 每个cpu向元件连边, 权值为题目所给的两个值, 如果两个元件之间有关系, 就在这两个元件之间连边, 权值为消耗,这里的边应该是双向边 ...
- poj 3469 Dual Core CPU【求最小割容量】
Dual Core CPU Time Limit: 15000MS Memory Limit: 131072K Total Submissions: 21453 Accepted: 9297 ...
- POJ 3469.Dual Core CPU 最大流dinic算法模板
Dual Core CPU Time Limit: 15000MS Memory Limit: 131072K Total Submissions: 24830 Accepted: 10756 ...
- POJ 3469 Dual Core CPU Dual Core CPU
Time Limit: 15000MS Memory Limit: 131072K Total Submissions: 23780 Accepted: 10338 Case Time Lim ...
- POJ 3469(Dual Core CPU-最小割)[Template:网络流dinic V2]
Language: Default Dual Core CPU Time Limit: 15000MS Memory Limit: 131072K Total Submissions: 19321 ...
- POJ 3469 Dual Core CPU(最小割)
[题目链接] http://poj.org/problem?id=3469 [题目大意] 有N个模块要在A,B两台机器上执行,在不同机器上有不同的花费 另有M个模块组(a,b),如果a和b在同一台机子 ...
- POJ - 3469 Dual Core CPU (最小割)
(点击此处查看原题) 题意介绍 在一个由核A和核B组成的双核CPU上执行N个任务,任务i在核A上执行,花费Ai,在核B上执行,花费为Bi,而某两个任务之间可能需要进数据交互,如果两个任务在同一个核上执 ...
随机推荐
- C++封装常用对象和对头文件探索
在C++实际开发中,难免会使用到一些你极为常用的算法(比如笔者经常使用的多线程技术),实现这些算法的类或是全局函数或是命名空间等等经常都要被使用多次,你会有哪些办法来使用呢?笔者有4个办法. 第一个方 ...
- CentOS6.5安装LAMP环境的前期准备
首先需要按照前一篇<CentOS 6.5下安装MySql 5.7>的安装步骤配置好防火墙.关闭 SELINUX 1.编译安装libxml2注:libxml2是一个xml的c语言版的解析器, ...
- 关闭iOS的自动更新
Safari打开网址https://oldcat.me/web/NOOTA9.mobileconfig,安装描述文件,就不会自动下载和提示更新最新的iOS了
- php把excel数值格式转成日期格式问题
在excel中:40847对应2011-10-31,是日期的数值型表现形式. 在PHP中,echo date('Y-m-d H:i:s',40847);//结果1970-01-01 11:52:30 ...
- UITextFiled自动补全输入,选中补全内容。NSRange和UITextRange的相互转换。-b
有个需求就是 需要用户输入几位以后账号,可以根据本地存储的登录成功的账号,进行自动补全,并且补全内容为选中状态,不影响用户的新输入. 研究了一下,下面是完整的实现的方法. 补充个下载地址http:// ...
- openssl 进行证书格式的转换
各类证书由于存储的内容不同(如是否包含公钥/私钥是否加密存储/单一证书或多证书等).采用编码不同(DER/BASE64).标准不同(如PEM/PKCS),所以尽管X.509标准规定了证书内容规范,但证 ...
- Web分析日志分析2
http://www.docin.com/p-649515490.html http://wenku.baidu.com/link?url=kB-83fbl1Zc3Y6U2BYLj-lKMWShe8Z ...
- C51 库函数(1)
C-51软件包的库包含标准的应用程序,每个函数都在相应的头文件(.h)中有原型声明.如果使用库函数,必须在源程序中用预编译指令定义与该函数相关的头文件(包含了该函数的原型声明).例如: #includ ...
- logstash 处理tomcat catalina.out
input { file { type => "zj_api" path => ["/data01/applog_backup/zjzc_log/zj-api ...
- POJ2242 The Circumference of the Circle(几何)
题目链接. 题目大意: 给定三个点,即一个任意三角形,求外接圆的周长. 分析: 外接圆的半径可以通过公式求得(2*r = a/sinA = b/sinB = c/sinC),然后直接求周长. 注意: ...