https://www.luogu.org/problemnew/show/P1854

题目描述

  某花店现有编号由 1 到 F 的 F 束花, 每一束花的品种都不一样. 编号由 1 到 V 的 V 个花瓶被按顺序摆成一行, 花瓶的数量至少和花的数量相同. 现在要把所有花放到花瓶中, 规定每个花瓶最多只能放一束花, 且 ∀ i < j, 第 i 束花必须放在第 j 束花的左侧.

  每个花瓶的形状和颜色不同, 因此放入不同的花会产生不同的美学效果. 空花瓶的美学值为 0. 请你给出一种方案, 使得总美学值最大.

输入输出格式

输入格式:

  第一行有两个整数 F 和 V, 分别为花束数和花瓶数 ( 1 ≤ F ≤ V ≤ 100 ).

  第二行到第 F + 1 行, 每行有 V 个整数. 第 i 行的第 j 个整数代表花束 i 放到花瓶 j 中的美学值.

输出格式:

  第一行输出一个整数,为最大美学值.

  第二行输出 F 个整数, 表示每束花放入的花瓶的编号.

输入输出样例

输入样例:

3 5

7 23 -5 -24 16

5 21 -4 10 23

-21 5 -4 -20 20

输出样例:

53

2 4 5

实现

  • 状态转移方程

  设 f[i][j] 为将编号 1 到 i 的花束放入编号 1 到 j 的花瓶中时最大的美学值, a[i][j]为将第 i 束花放入第 j 个花瓶所能产生的美学值, 有 f[i][j] = max(f[i - 1][j - 1] + a[i][j], f[i][j - 1]).

  • 滚动数组

  可以省去表示花束的一维, 用 f[i] 表示在前 i 个花瓶中放入当前数量花束的最大美学值.

  例如, 对于样例, 可以按如下方法计算:

  初始状态:

i 0 1 2 3 4 5
f[i] 0 0 0 0 0 0

  考虑第一束花后:

i 0 1 2 3 4 5
f[i] 0 7 23 -5 -24 16

  ∵ 此时 f[i] 表示在前 i 个花瓶中放入第一束花的最大美学值.

  ∴ ∀ i > j, f[i] ≥ f[j], f 数组应修改为:

i 0 1 2 3 4 5
f[i] 0 7 23 23 23 23

  以此类推, 考虑第二束花:

i 0 1 2 3 4 5
f[i] 0 0 28 19 33 46
i 0 1 2 3 4 5
f[i] 0 0 28 28 33 46

  考虑第三束花:

i 0 1 2 3 4 5
f[i] 0 0 0 24 8 53
i 0 1 2 3 4 5
f[i] 0 0 0 24 24 53

  由此得到答案 53. 显然, 题目所要求的摆放方案可以在刷表的同时记录下来.

实现

 #include <iostream>
 #include <cstdio>
 #define IsDigit(x) ((x) >= '0' && (x) <= '9')
 using namespace std;

 int f, v;
 ][], dp[], plan[][][], s[];

 int Read(void)
 {
     ), sign(false);

     c = getchar();
     while (!IsDigit(c) && c != '-')
         c = getchar();
     c == '-' && (sign = true, c = getchar());
     do
         ret = ret *  + c - ';
     while ((c = getchar()) && IsDigit(c));
     return sign ? -ret : ret;
 }

 int main()
 {
     ), nxt;

     f = Read();
     v = Read();
     ; i <= f; ++i)
         ; j <= v; ++j)
             in[i][j] = Read();
     ; i <= f; ++i) {
         for (int j = v - f + i; j >= i; --j) {
             dp[j] = dp[j - ] + in[i][j];
             plan[i][j][] = plan[i - ][j - ][];
             plan[i][j][] = j;
         }
         ; j <= v - f + i; ++j)
             dp[j - ] > dp[j] &&
              (dp[j] = dp[j - ], plan[i][j][] = plan[i][j - ][], plan[i][j][] = plan[i][j - ][]);
     }
     nxt = v;
     for (int i = f; i; --i) {
         s[pos++] = plan[i][nxt][];
         nxt = plan[i][nxt][];
     }
     printf("%d\n", dp[v]);
     ; i >= ; --i)
         printf("%d ", s[i]);
     printf("\n");
     ;
 }

洛谷【P1854】花店橱窗布置的更多相关文章

  1. 洛谷P1854 花店橱窗布置 分析+题解代码

    洛谷P1854 花店橱窗布置 分析+题解代码 蒟蒻的第一道提高+/省选-,纪念一下. 题目描述: 某花店现有F束花,每一束花的品种都不一样,同时至少有同样数量的花瓶,被按顺序摆成一行,花瓶的位置是固定 ...

  2. 洛谷 P1854 花店橱窗布置 【dp】

    题目描述 某花店现有F束花,每一束花的品种都不一样,同时至少有同样数量的花瓶,被按顺序摆成一行,花瓶的位置是固定的,从左到右按1到V顺序编号,V是花瓶的数目.花束可以移动,并且每束花用1到F的整数标识 ...

  3. 洛谷 P1854 花店橱窗布置

    题目描述 某花店现有F束花,每一束花的品种都不一样,同时至少有同样数量的花瓶,被按顺序摆成一行,花瓶的位置是固定的,从左到右按1到V顺序编号,V是花瓶的数目.花束可以移动,并且每束花用1到F的整数标识 ...

  4. 洛谷P1854 花店橱窗布置

    题目 DP,直接递推比记忆化搜索简单. 定义状态\(dp[i][j]\)为前i行最后一个选择第i行第j个数所得到最大值. 易得状态转移方程 \(dp[i][j]=max(dp[i-1][k]+a[i] ...

  5. 洛谷 P1854 花店橱窗布置 题解

    Analysis 给定一个f*v的矩阵 要求从第一行走到第f行,每行取走一个数, 且该行所取的数必须必上一行所取的数的列数大 , 求所能取走的最大值 注意每一行所取走的数字的列数必须大于等该行的行号 ...

  6. [动态规划]P1854 花店橱窗布置

    题目描述 某花店现有F束花,每一束花的品种都不一样,同时至少有同样数量的花瓶,被按顺序摆成一行,花瓶的位置是固定的,从左到右按1到V顺序编号,V是花瓶的数目.花束可以移动,并且每束花用1到F的整数标识 ...

  7. luogu P1854 花店橱窗布置

    题目描述 某花店现有F束花,每一束花的品种都不一样,同时至少有同样数量的花瓶,被按顺序摆成一行,花瓶的位置是固定的,从左到右按1到V顺序编号,V是花瓶的数目.花束可以移动,并且每束花用1到F的整数标识 ...

  8. 题解 P1854 花店橱窗布置

    把二维压成一维的DP了解一下... 传送门 (以纪念神经兮兮调了两天的一维DP(刷水题谋财害命)以及感谢学长的帮助@ydnhaha) 显然我们有二维的dp:f[i][j]代表第i盆花放到第j个位置 ; ...

  9. 【Luogu】P1854花店橱窗布置(DP)

    照例良心题目链接 此题使用f[i][j]表示前i束花放进前j个花瓶的时候的最大值.转移方程如下 f[i][j]=max(f[i][j-1],f[i-1][j-1]+que[i][j]) 其中que[i ...

  10. [IOI1999]花店橱窗布置(DP路径记录)

    题目:[IOI1999]花店橱窗布置 问题编号:496 题目描述 某花店现有F束花,每一束花的品种都不一样,同时至少有同样数量的花瓶,被按顺序摆成一行,花瓶的位置是固定的,从左到右按1到V顺序编号,V ...

随机推荐

  1. SFTP 服务搭建

    1. 介绍 sftp是Secure File Transfer Protocol的缩写,安全文件传送协议.可以为传输文件提供一种安全的加密方法.sftp 与 ftp 有着几乎一样的语法和功能.SFTP ...

  2. [iOS]CIDetector之CIDetectorTypeFace人脸识别

    - (void)viewDidLoad { [super viewDidLoad]; // Do any additional setup after loading the view, typica ...

  3. 【模板】区间dp

    有N堆石子排成一排,每堆石子有一定的数量.现要将N堆石子合并为1堆.在合并的过程中只能每次将相邻的两堆石子合并,每次合并的花费为这两堆石子之和,求合并成1堆的最小花费. dp[i][j]表示将区间[i ...

  4. linux下安装protobuf及cmake编译

    一.protobuf 安装 protobuf版本:2.6.1 下载地址:https://github.com/google/protobuf/archive/v2.6.1.zip 解压之后进入目录 修 ...

  5. 微信小程序实现转义换行符

    在html中可以直接使用<br />换行,但是小程序wxml中使用<br />无效,可以换成\n Page({ data: { title: '至少5个字\n请多说些感受吧' ...

  6. webpack4+Vue搭建自己的Vue-cli

    前言 最近在看webpack4,深感知识浅薄,这两天也一直在思考cli的配置,借助一些别人的实践,尝试自己搭建vue的项目,这里使用webpack4版本,之前我在网上查找别人的vue项目搭建,但是都是 ...

  7. python学习笔记:第9天 函数初步

    1. 函数的定义及调用 函数:所谓的函数可以看作是对一段代码的封装,也是对一个功能模块的封装,这样方便在下次想用这个功能的时候直接调用这个功能模块,而不用重新去写. 函数的定义:我们使用def关键字来 ...

  8. Jquery 批量操作标签属性

     $("[id*='Custom']").removeAttr("disabled")

  9. Debian使用dpkg安装MySQL

    说明 本文写于2017-10-03,使用MySQL 5.7,操作系统为64位 Debian GNU/Linux 8.6 (jessie). 安装 因apt仓库将mysql相关的包移除,需要自己去官网下 ...

  10. 20155207王雪纯 《Java程序设计》实验一报告

    20155207王雪纯 <Java程序设计>实验一报告 课程:Java程序设计 班级:1552 指导教师:娄嘉鹏 实验日期:2017.04.07 实验名称:Java开发环境的熟悉(Linu ...