COGS738 [网络流24题] 数字梯形(最小费用最大流)
题目这么说:
给定一个由n 行数字组成的数字梯形如下图所示。梯形的第一行有m 个数字。从梯形
的顶部的m 个数字开始,在每个数字处可以沿左下或右下方向移动,形成一条从梯形的顶
至底的路径。
规则1:从梯形的顶至底的m条路径互不相交。
规则2:从梯形的顶至底的m条路径仅在数字结点处相交。
规则3:从梯形的顶至底的m条路径允许在数字结点相交或边相交。
对于给定的数字梯形,分别按照规则1,规则2,和规则3 计算出从梯形的顶至底的m
条路径,使这m条路径经过的数字总和最大。
对那三种情况分别建容量网络跑最小费用最大流,这个很容易的。。要注意题意上说的是顶部m个数字分别作为m条路径的出发点。
#include<cstdio>
#include<cstring>
#include<queue>
#include<algorithm>
using namespace std;
#define INF (1<<30)
#define MAXN 888
#define MAXM 888*888*2
struct Edge{
int u,v,cap,cost,next;
}edge[MAXM];
int head[MAXN];
int NV,NE,vs,vt; void addEdge(int u,int v,int cap,int cost){
edge[NE].u=u; edge[NE].v=v; edge[NE].cap=cap; edge[NE].cost=cost;
edge[NE].next=head[u]; head[u]=NE++;
edge[NE].u=v; edge[NE].v=u; edge[NE].cap=; edge[NE].cost=-cost;
edge[NE].next=head[v]; head[v]=NE++;
}
bool vis[MAXN];
int d[MAXN],pre[MAXN];
bool SPFA(){
for(int i=;i<NV;++i){
vis[i]=; d[i]=INF;
}
vis[vs]=; d[vs]=;
queue<int> que;
que.push(vs);
while(!que.empty()){
int u=que.front(); que.pop();
for(int i=head[u]; i!=-; i=edge[i].next){
int v=edge[i].v;
if(edge[i].cap && d[v]>d[u]+edge[i].cost){
d[v]=d[u]+edge[i].cost;
pre[v]=i;
if(!vis[v]){
vis[v]=;
que.push(v);
}
}
}
vis[u]=;
}
return d[vt]!=INF;
}
int MCMF(){
int res=;
while(SPFA()){
int flow=INF,cost=;
for(int u=vt; u!=vs; u=edge[pre[u]].u){
flow=min(flow,edge[pre[u]].cap);
}
for(int u=vt; u!=vs; u=edge[pre[u]].u){
edge[pre[u]].cap-=flow;
edge[pre[u]^].cap+=flow;
cost+=flow*edge[pre[u]].cost;
}
res+=cost;
}
return res;
}
int a[][],idx[][];
int main(){
freopen("digit.in","r",stdin);
freopen("digit.out","w",stdout);
int m,n;
scanf("%d%d",&m,&n);
int cnt=;
for(int i=; i<n; ++i){
for(int j=; j<m+i; ++j) scanf("%d",&a[i][j]),idx[i][j]=cnt++;
}
vs=cnt*; vt=vs+; NV=vt+; NE=;
memset(head,-,sizeof(head));
for(int i=; i<n; ++i){
for(int j=; j<m+i; ++j){
addEdge(idx[i][j],idx[i][j]+cnt,,-a[i][j]);
if(i==) addEdge(vs,idx[i][j],,);
if(i==n-) addEdge(idx[i][j]+cnt,vt,,);
else{
addEdge(idx[i][j]+cnt,idx[i+][j],,);
addEdge(idx[i][j]+cnt,idx[i+][j+],,);
}
}
}
printf("%d\n",-MCMF()); vs=cnt*; vt=vs+; NV=vt+; NE=;
memset(head,-,sizeof(head));
for(int i=; i<n; ++i){
for(int j=; j<m+i; ++j){
addEdge(idx[i][j],idx[i][j]+cnt,INF,-a[i][j]);
if(i==) addEdge(vs,idx[i][j],,);
if(i==n-) addEdge(idx[i][j]+cnt,vt,INF,);
else{
addEdge(idx[i][j]+cnt,idx[i+][j],,);
addEdge(idx[i][j]+cnt,idx[i+][j+],,);
}
}
}
printf("%d\n",-MCMF()); vs=cnt*; vt=vs+; NV=vt+; NE=;
memset(head,-,sizeof(head));
for(int i=; i<n; ++i){
for(int j=; j<m+i; ++j){
addEdge(idx[i][j],idx[i][j]+cnt,INF,-a[i][j]);
if(i==) addEdge(vs,idx[i][j],,);
if(i==n-) addEdge(idx[i][j]+cnt,vt,INF,);
else{
addEdge(idx[i][j]+cnt,idx[i+][j],INF,);
addEdge(idx[i][j]+cnt,idx[i+][j+],INF,);
}
}
}
printf("%d",-MCMF());
return ;
}
COGS738 [网络流24题] 数字梯形(最小费用最大流)的更多相关文章
- LIbreOJ #6011. 「网络流 24 题」运输问题 最小费用最大流
		
#6011. 「网络流 24 题」运输问题 内存限制:256 MiB时间限制:1000 ms标准输入输出 题目类型:传统评测方式:文本比较 上传者: 匿名 提交提交记录统计讨论测试数据 题目描述 ...
 - CGOS461 [网络流24题] 餐巾(最小费用最大流)
		
题目这么说的: 一个餐厅在相继的N天里,第i天需要Ri块餐巾(i=l,2,…,N).餐厅可以从三种途径获得餐巾. 购买新的餐巾,每块需p分: 把用过的餐巾送到快洗部,洗一块需m天,费用需f分(f< ...
 - 网络流24题——数字梯形问题 luogu 4013
		
题目描述:这里 极其裸的一道费用流问题 首先分析第一问,由于要求一个点只能经过一次,所以我们将梯形中的每一个点拆成两个点(记为入点和出点,顾名思义,入点用来承接上一行向这一行的边,出点用来向下一行连边 ...
 - 【网络流#2】hdu 1533 - 最小费用最大流模板题
		
最小费用最大流,即MCMF(Minimum Cost Maximum Flow)问题 嗯~第一次写费用流题... 这道就是费用流的模板题,找不到更裸的题了 建图:每个m(Man)作为源点,每个H(Ho ...
 - 【PowerOJ1752&网络流24题】运输问题(费用流)
		
题意: 思路: [问题分析] 费用流问题. [建模方法] 把所有仓库看做二分图中顶点Xi,所有零售商店看做二分图中顶点Yi,建立附加源S汇T. 1.从S向每个Xi连一条容量为仓库中货物数量ai,费用为 ...
 - Cogs 731. [网络流24题] 最长递增子序列(最大流)
		
[网络流24题] 最长递增子序列 ★★★☆ 输入文件:alis.in 输出文件:alis.out 简单对比 时间限制:1 s 内存限制:128 MB «问题描述: 给定正整数序列x1,-, xn. ( ...
 - 2018.10.14 loj#6003. 「网络流 24 题」魔术球(最大流)
		
传送门 网络流好题. 这道题可以动态建图. 不难想到把每个球iii都拆点成i1i_1i1和i2i_2i2,每次连边(s,i1),(i2,t)(s,i_1),(i_2,t)(s,i1),(i2, ...
 - 2018.10.14 loj#6012. 「网络流 24 题」分配问题(费用流)
		
传送门 费用流水题. 依然是照着题意模拟建边就行了. 为了练板子又重新写了一遍费用流. 代码: #include<bits/stdc++.h> #define N 305 #define ...
 - 2018.10.14 loj#6011. 「网络流 24 题」运输问题(费用流)
		
传送门 费用流入门题. 直接按照题意模拟. 把货物的数量当做容量建边. 然后跑一次最小费用流和最大费用流就行了. 代码: #include<bits/stdc++.h> #define N ...
 
随机推荐
- 3ds max的动画输出
			
转自:http://zhidao.baidu.com/link?url=qc3vV2A9-ydb-YiVKoF7z_bIIRlmLSkyl8DcuWNYn8FaBxa2BDVLwuGPX_jYWxbw ...
 - 昨天在公司加班,上午好像就是弄一个ftp的linux服务问题
			
在网上找了一些方法,可是其中有通过匿名方式登陆,但是在root的权限下才能存放文件,可是把匿名用户登陆取消之后又不能登陆,就是没有列出怎么来添加一个ftp的用户,今天打算直接装一个linux系统在虚拟 ...
 - vim 打开高亮和关闭高亮
			
:set hls 找开高亮 :set nohls 关闭高亮
 - 关闭 Visual Studio 2013 的 Browser Link 功能
			
最近公司弄新项目需要用 MVC,就把 IDE 升级到了 Visual Studio 2013,在开发的时候发现有好多请求一个本地49925的端口 . 很奇怪,一开始以为是 Visual Studio ...
 - 菜鸟学Linux命令:tar命令 压缩与解压缩
			
tar命令可以为linux的文件和目录创建档案.利用tar,可以为某一特定文件创建档案(备份文件),也可以在档案中改变文件,或者向档案中加入新的文件. tar最初被用来在磁带上创建档案,现在,用户可以 ...
 - 设计模式学习之策略模式(Strategy,行为型模式)(13)
			
转载地址:http://www.cnblogs.com/zhili/p/StragetyPattern.html 一.引言 本文要介绍的策略模式也就是对策略进行抽象,策略的意思就是方法,所以也就是对方 ...
 - 基于类和基于函数的python多线程样例
			
不断的练,加深记忆吧. #!/usr/bin/env python # -*- coding: utf-8 -*- import threading import time exitFlag = 0 ...
 - 攻城狮在路上(叁)Linux(二十六)--- linux文件系统的特殊查看与操作
			
一.boot sector 与 super block的关系: 1.boot sector用于存放引导装载程序,占用1024个字节. 2.super block的大小也为1024字节. 3.若bloc ...
 - Oracle Segments可以跨多个data files吗?
			
首先,你需要明白的一点是:数据库的物理结构是由数据库的操作系统文件所决定,每一个Oracle数据库是由三种类型的文件组成:数据文件.日志文件和控制文件.数据库的文件为数据库信息提供真正的物理存储.每一 ...
 - PHP json数据格式化方法
			
php 的json_encode能把数组转换为json格式的字符串.字符串没有缩进,中文会转为unicode编码,例如\u975a\u4ed4.人阅读比较困难.现在这个方法在json_encode的基 ...
 
			
		