数字梯形(cogs 738)
«问题描述:
给定一个由n 行数字组成的数字梯形如下图所示。梯形的第一行有m 个数字。从梯形
的顶部的m 个数字开始,在每个数字处可以沿左下或右下方向移动,形成一条从梯形的顶
至底的路径。
规则1:从梯形的顶至底的m条路径互不相交。
规则2:从梯形的顶至底的m条路径仅在数字结点处相交。
规则3:从梯形的顶至底的m条路径允许在数字结点相交或边相交。
«编程任务:
对于给定的数字梯形,分别按照规则1,规则2,和规则3 计算出从梯形的顶至底的m
条路径,使这m条路径经过的数字总和最大。
«数据输入:
由文件digit.in提供输入数据。文件的第1 行中有2个正整数m和n(m,n<=20),分别
表示数字梯形的第一行有m个数字,共有n 行。接下来的n 行是数字梯形中各行的数字。
第1 行有m个数字,第2 行有m+1 个数字,…。
«结果输出:
程序运行结束时,将按照规则1,规则2,和规则3 计算出的最大数字总和输出到文件
digit.out中。每行一个最大总和。
输入文件示例 输出文件示例
digit.in
2 5
2 3
3 4 5
9 10 9 1
1 1 10 1 1
1 1 10 12 1 1
digit.out
66
75
77
- /*
- 第一个建图就是拆点(保证每个点只走一次),第二个建图是把两个点之间的边设为1,第三个inf随意搞。
- */
- #include<cstdio>
- #include<iostream>
- #include<queue>
- #include<cstring>
- #define N 4010
- #define inf 1000000000
- using namespace std;
- int a[N][N],head[N],dis[N],inq[N],fa[N],n,m,num,cnt,S,T;
- struct node{int u,v,pre,f,w;}e[N];
- void add(int u,int v,int f,int w){
- e[++cnt].u=u;e[cnt].v=v;e[cnt].f=f;e[cnt].w=w;e[cnt].pre=head[u];head[u]=cnt;
- e[++cnt].u=v;e[cnt].v=u;e[cnt].f=;e[cnt].w=-w;e[cnt].pre=head[v];head[v]=cnt;
- }
- bool spfa(){
- for(int i=;i<=T;i++) dis[i]=inf;
- queue<int> q;q.push(S);inq[S]=;dis[S]=;
- while(!q.empty()){
- int u=q.front();q.pop();inq[u]=;
- for(int i=head[u];i;i=e[i].pre)
- if(e[i].f&&dis[e[i].v]>dis[u]+e[i].w){
- dis[e[i].v]=dis[u]+e[i].w;
- fa[e[i].v]=i;
- if(!inq[e[i].v]){
- inq[e[i].v]=;
- q.push(e[i].v);
- }
- }
- }
- return dis[T]!=inf;
- }
- void mincost(){
- int cost=;
- while(spfa()){
- int tmp=fa[T],x=inf;
- while(tmp){
- int u=e[tmp].u;
- x=min(x,e[tmp].f);
- tmp=fa[e[tmp].u];
- }
- tmp=fa[T];
- while(tmp){
- e[tmp].f-=x;
- e[tmp^].f+=x;
- tmp=fa[e[tmp].u];
- }
- cost+=x*dis[T];
- }
- printf("%d\n",-cost);
- }
- int hao(int i,int j){
- return (m*+i-)*(i-)/+j;
- }
- void build1(){
- cnt=;memset(head,,sizeof(head));
- for(int i=;i<=m;i++)
- add(S,i,,-a[][i]);
- for(int i=;i<n;i++)
- for(int j=;j<=m+i-;j++)
- add(hao(i,j)+num,hao(i+,j),,-a[i+][j]),add(hao(i,j)+num,hao(i+,j+),,-a[i+][j+]);
- for(int i=;i<=m+n-;i++)
- add(hao(n,i)+num,T,,);
- for(int i=;i<=n;i++)
- for(int j=;j<=m+i-;j++)
- add(hao(i,j),hao(i,j)+num,,);
- }
- void build2(){
- cnt=;memset(head,,sizeof(head));
- for(int i=;i<=m;i++)
- add(S,i,,-a[][i]);
- for(int i=;i<=n;i++)
- for(int j=;j<=m+i-;j++)
- add(hao(i,j),hao(i+,j),,-a[i+][j]),add(hao(i,j),hao(i+,j+),,-a[i+][j+]);
- for(int i=;i<=m+n-;i++)
- add(hao(n,i),T,inf,);
- }
- void build3(){
- cnt=;memset(head,,sizeof(head));
- for(int i=;i<=m;i++)
- add(S,i,,-a[][i]);
- for(int i=;i<=n;i++)
- for(int j=;j<=m+i-;j++)
- add(hao(i,j),hao(i+,j),inf,-a[i+][j]),add(hao(i,j),hao(i+,j+),inf,-a[i+][j+]);
- for(int i=;i<=m+n-;i++)
- add(hao(n,i),T,inf,);
- }
- int main(){
- scanf("%d%d",&m,&n);num=(m*+n-)*n/;
- S=;T=num*+;
- for(int i=;i<=n;i++)
- for(int j=;j<=m+i-;j++)
- scanf("%d",&a[i][j]);
- build1();mincost();
- build2();mincost();
- build3();mincost();
- return ;
- }
数字梯形(cogs 738)的更多相关文章
- COGS738 [网络流24题] 数字梯形(最小费用最大流)
题目这么说: 给定一个由n 行数字组成的数字梯形如下图所示.梯形的第一行有m 个数字.从梯形的顶部的m 个数字开始,在每个数字处可以沿左下或右下方向移动,形成一条从梯形的顶至底的路径.规则1:从梯形的 ...
- 【wikioi】1913 数字梯形问题(费用流)
http://wikioi.com/problem/1913/ 如果本题没有询问2和3,那么本题和蚯蚓那题一模一样.http://www.cnblogs.com/iwtwiioi/p/3935039. ...
- 【网络流24题】No.16 数字梯形问题 (不相交路径 最大费用流)
[题意] 给定一个由 n 行数字组成的数字梯形如下图所示. 梯形的第一行有 m 个数字.从梯形的顶部的 m 个数字开始,在每个数字处可以沿左下或右下方向移动, 形成一条从梯形的顶至底的路径.规则 1: ...
- codevs 1913 数字梯形问题 费用流
题目链接 给你一个数字梯形, 最上面一层m个数字, 然后m+1,......m+n-1个. n是层数. 在每个位置, 可以向左下或右下走.然后让你从最顶端的m个数字开始, 走出m条路径, 使得路过的数 ...
- P4013 数字梯形问题 网络流
题目描述 给定一个由 nn 行数字组成的数字梯形如下图所示. 梯形的第一行有 mm 个数字.从梯形的顶部的 mm 个数字开始,在每个数字处可以沿左下或右下方向移动,形成一条从梯形的顶至底的路径. 分别 ...
- 【刷题】LOJ 6010 「网络流 24 题」数字梯形
题目描述 给定一个由 \(n\) 行数字组成的数字梯形如下图所示.梯形的第一行有 \(m\) 个数字.从梯形的顶部的 \(m\) 个数字开始,在每个数字处可以沿左下或右下方向移动,形成一条从梯形的顶至 ...
- Libre 6010「网络流 24 题」数字梯形 (网络流,最大费用最大流)
Libre 6010「网络流 24 题」数字梯形 (网络流,最大费用最大流) Description 给定一个由n 行数字组成的数字梯形如下图所示.梯形的第一行有m 个数字.从梯形的顶部的m 个数字开 ...
- P4013 数字梯形问题 网络流二十四题
P4013 数字梯形问题 题目描述 给定一个由 nn 行数字组成的数字梯形如下图所示. 梯形的第一行有 m 个数字.从梯形的顶部的 m 个数字开始,在每个数字处可以沿左下或右下方向移动,形成一条从梯形 ...
- 【费用流】【网络流24题】【P4013】 数字梯形问题
Description 给定一个由 \(n\) 行数字组成的数字梯形如下图所示. 梯形的第一行有 \(m\) 个数字.从梯形的顶部的 \(m\) 个数字开始,在每个数字处可以沿左下或右下方向移动,形成 ...
随机推荐
- 新建maven的pom.xml第一行出错的解决思路
前言:博主在想要用maven创建项目的时候,忘记之前已经安装过maven了,所以再安装了另一个版本的maven,导致在pom.xml的第一行总是显示某一个jar的zip文件读取不出来. 在网上找了很多 ...
- shiro 配置拦截规则之后css和js等失效
使用shiro作为平台的权限管理工具,shiro的配置文件如下: package com.ros.config; import java.util.LinkedHashMap;import java. ...
- 历史管理 onhashchange
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/ ...
- python基础一 day14 复习
迭代器和生成器迭代器:双下方法 : 很少直接调用的方法.一般情况下,是通过其他语法触发的可迭代的 —— 可迭代协议 含有__iter__的方法('__iter__' in dir(数据))可迭代的一定 ...
- node mocha mochawesome报安装不成功
1.进行cnpm安装: npm install cnpm -g --registry=https://registry.npm.taobao.org 2.查看cnpm版本 cnpm -v 3.安装mo ...
- java 去掉html/style/css等标签
//定义script的正则表达式 private static String regEx_script="<script[^>]*?>[\\s\\S]*?<\\/sc ...
- 安装vc++6.0的步骤
我们学习计算机,就必须要先将编程的c语言学好,打好基础,学习c语言最好的方法就是多上机联系,对于联系我们需要在自己的电脑上安装vc++6.0来进行平日里的联系.1.打开电脑进行联网,打开浏览器搜索vc ...
- CF-1099 D. Sum in the tree
CF-1099 D. Sum in the tree 题意:结点序号为 1~n 的一个有根树,根序号为1,每个点有一个权值a[i], 然后定义一s[i]表示从根节点到 结点序号为i的结点的路途上所经过 ...
- HDU-2544-最短路(Bellman-Ford)
Bellman-Ford算法是一个时间复杂度很高,但是它可以用来判断负环 负环就是上面的图,那个环的整体值小于零了,所以就是负环. 我们用Bellman-Ford算法进行更新,打一个表出来: k a ...
- Linux菜鸟起飞之路【四】绝对路径、相对路径及常用目录
一.绝对路径与相对路径 Linux操作系统中存在着两种路径:绝对路径和相对路径.我们在访问文件或文件夹的时候,其实都是通过路径来操作的.两种路径在实际操作中能起到同等的作用. 在开始具体介绍之前,我们 ...