Description

KC来到了一个盛产瓷器的国度。他来到了一位商人的店铺。在这个店铺中,KC看到了一个有n(1<=n<=100)排的柜子,每排都有一些瓷器,每排不超过100个。那些精美的艺术品使KC一下心动了,决定从N排的商品中买下m(1<=m<=10000)个瓷器。

这个商人看KC的脸上长满了痘子,就像苔藓一样,跟精美的瓷器相比相差太多,认为这么精致的艺术品被这样的人买走艺术价值会大打折扣。商人感到不爽,于是规定每次取商品只能取其中一排的最左边或者最右边那个,想为难KC。

现在KC又获知每个瓷器的价值(用一个不超过100的正整数表示),他希望取出的m个商品的总价值最大。

Input

输入文件的第一行包括两个正整数n,m;

接下来2到n+1行,第i行第一个数表示第i排柜子的商品数量Si,接下来Si个数表示从左到右每个商品的价值。

Output

输出文件只有一个正整数,即m个商品最大的总价值。

Sample Input

输入1:

2 3

3 3 7 2

3 4 1 5

输入2:

1 3

4 4 3 1 2

Sample Output

输出1:

15

样例解释1:

取第一排的最左边两个和第二排的最右边那个。总价直为3+7+5=15;

输出2:

9

Data Constraint

对于10%的数据,Si=1,1<=i<=nS_i=1,1<=i<=nSi​=1,1<=i<=n。

对于另外10%的数据,n=1n=1n=1.

Analysis

  • 有nnn行,n∈[1,100]n\in[1,100]n∈[1,100],每行个数有可能不同。
  • 一共购买mmm个物品,m∈[1,10000]m\in[1,10000]m∈[1,10000]
  • 每行可以购买前iii个物品和后jjj个物品,i,j∈[0,100]i,j \in [0,100]i,j∈[0,100]

Solution

容易发现,每行的最优都可以单独预处理算出,而当每行购买iii个商品的最大价值确定后,就可以转换为一个多重背包问题。

于是预处理每一行的购买iii件商品的最大价值。

设在该行一共有s个商品,购买k个商品,在左边购买m个商品,则在右边购买k−m个商品设在该行一共有s个商品,购买k个商品,在左边购买m个商品,则在右边购买k-m个商品设在该行一共有s个商品,购买k个商品,在左边购买m个商品,则在右边购买k−m个商品

容易得到:

val[k]=∑m=0k(∑i=1mvali+∑j=s−(k−m)+1svalj)val[k] = \sum_{m=0}^k (\sum_{i=1}^m val_i +\sum_{j=s-(k-m)+1}^s val_j)val[k]=∑m=0k​(∑i=1m​vali​+∑j=s−(k−m)+1s​valj​)

当然,区间和我们可以用简单的前缀和算出:

for(int i = 1;i<=s;++i)//处理前缀和为以后求区间和做准备
sum[i] = sum[i-1] + val[i];

于是上式可以简化成:

设s个商品在左边取了m个,在右边取了k−m个设s个商品在左边取了m个,在右边取了k-m个设s个商品在左边取了m个,在右边取了k−m个

val[k]=summ+sums−sums−(k−m)val[k] = sum_m + sum_s - sum_{s-(k-m)}val[k]=summ​+sums​−sums−(k−m)​

前缀和求区间和:sum[i,j]=sumj−sumi−1sum_{[i,j]}=sum_j-sum_{i-1}sum[i,j]​=sumj​−sumi−1​

由于数据量小,可以直接枚举这个m,得到此时的最大价值。

那么我们就可以用O(s)的复杂度预处理,随后用O(s2s^2s2)的复杂度求出该商品购买所有件数时的最大价值。

其中s∈[1,100]s\in[1,100]s∈[1,100],有nnn种商品,n∈[1,100]n\in[1,100]n∈[1,100],每种都需要O(s2)O(s^2)O(s2)进行预处理,复杂度可能达到O(ns2)O(ns^2)O(ns2)但由于常数很小,还是可以接受的。

	for(register int k = 1;k<=n;++k)//第k种
for(register int i = 1;i<=s[k][0];++i)
//表示取i个的时候,s[k][0]代表该商品总个数
for(register int j = 0;j<=i;++j)
dp[k][i] = MAX(dp[k][i],\
sum[k][j] + sum[k][s[k][0]] - sum[k][s[k][0] - i + j]);

接下来就是多重背包问题了:

有n种商品,每种商品有pi个,第i种商品取j个得到的最大价值为sij,有n种商品,每种商品有p_i个,第i种商品取j个得到的最大价值为s_{ij},有n种商品,每种商品有pi​个,第i种商品取j个得到的最大价值为sij​,

每个商品的体积都为1,容量为m,问最多能取得多大价值。每个商品的体积都为1,容量为m,问最多能取得多大价值。每个商品的体积都为1,容量为m,问最多能取得多大价值。

	//dp[i][j]代表在第i行取j个商品的最大获利
//f[i][j]代表在前i行取j个商品
for(register int i = 1;i<=n;++i) //在第i行选商品
for(register int j = 1;j<=m;++j)//此时一共选择j个商品
for(register int k = 0;k<=s[i][0] && k <= j;++k)
{
//k是在这一行取得多少商品,注意k不能超过当前取商品量
//当前在前i行取j个商品,在当前行取k个商品,那么就在前i-1行取j-k个商品
dp[i][j] = MAX(dp[i][j],dp[i-1][j-k] + f[i][k]);
}

本题结束。

Code

#include <cstdio>
#define MAX(x,y) ((x)>(y)?(x):(y))
using namespace std;
void read(int &r)
{
static char c;
for(c=getchar();c>'9'||c<'0';c=getchar());
for(;c<='9'&&c>='0';r=(r<<1)+(r<<3)+c-48,c=getchar());
}
int n,m;
int s[105][105];
int dp[105][10005];//表示从第i排取出j个的最大价值和
int f[105][105];
int sum[105][105];
int main()
{
read(n);
read(m);
for(register int i = 1;i<=n;++i)
{
read(s[i][0]);
for(register int j = 1;j<=s[i][0];++j)
read(s[i][j]);
} for(register int i = 1;i<=n;++i)//处理前缀和为以后求区间和做准备
for(register int j = 1;j<=s[i][0];++j)
sum[i][j] = sum[i][j-1] + s[i][j]; //每行按行dp算出取n个商品时的最大值
for(register int k = 1;k<=n;++k)
for(register int i = 1;i<=s[k][0];++i) //表示取i个的时候
for(register int j = 0;j<=i;++j)
f[k][i] = MAX(f[k][i],sum[k][j] + sum[k][s[k][0]] - sum[k][s[k][0] - i + j]);
//在左边取j个,右边取 i - j个
//使用前缀和的方法求区间 //f[i][j]代表在第i行取j个商品的最大获利
//dp[i][j]代表在前i行取j个商品
for(register int i = 1;i<=n;++i) //在第i行选商品
for(register int j = 1;j<=m;++j)//此时一共选择j个商品
for(register int k = 0;k<=s[i][0] && k <= j;++k)
{
//k是在这一行取得多少商品,注意k不能超过当前取商品量
//当前在前i行取j个商品,在当前行取k个商品,那么就在前i-1行取j-k个商品
dp[i][j] = MAX(dp[i][j],dp[i-1][j-k] + f[i][k]);
}
printf("%d",dp[n][m]);
return 0;
}

[JZOJ]3413.KC的瓷器的更多相关文章

  1. noip2018 pre——Dp

    Dp专题 1011: KC的瓷器 (porcelain) 题目描述 KC来到了一个盛产瓷器的国度.他来到了一位商人的店铺.在这个店铺中,KC看到了一个有n(1<=n<=100)排的柜子,每 ...

  2. (jzoj snow的追寻)线段树维护树的直径

    jzoj snow的追寻 DFS序上搞 合并暴力和,记录最长链和当前最远点,距离跑LCA # include <stdio.h> # include <stdlib.h> # ...

  3. [jzoj]3506.【NOIP2013模拟11.4A组】善良的精灵(fairy)(深度优先生成树)

    Link https://jzoj.net/senior/#main/show/3506 Description 从前有一个善良的精灵. 一天,一个年轻人B找到她并请他预言他的未来.这个精灵透过他的水 ...

  4. [jzoj]3468.【NOIP2013模拟联考7】OSU!(osu)

    Link https://jzoj.net/senior/#main/show/3468 Description osu 是一款群众喜闻乐见的休闲软件. 我们可以把osu的规则简化与改编成以下的样子: ...

  5. [jzoj]5478.【NOIP2017提高组正式赛】列队

    Link https://jzoj.net/senior/#main/show/5478 Description Sylvia 是一个热爱学习的女孩子.       前段时间,Sylvia 参加了学校 ...

  6. [jzoj]1115.【HNOI2008】GT考试

    Link https://jzoj.net/senior/#main/show/1115 Description 申准备报名参加GT考试,准考证号为n位数X1X2X3...Xn-1Xn(0<=X ...

  7. [jzoj]2538.【NOIP2009TG】Hankson 的趣味题

    Link https://jzoj.net/senior/#main/show/2538 Description Hanks 博士是BT (Bio-Tech,生物技术) 领域的知名专家,他的儿子名叫H ...

  8. [jzoj]4216.【NOIP2015模拟9.12】平方和

    Link https://jzoj.net/senior/#main/show/4216 Description 给出一个N个整数构成的序列,有M次操作,每次操作有一下三种: ①Insert Y X, ...

  9. [jzoj]2938.【NOIP2012模拟8.9】分割田地

    Link https://jzoj.net/senior/#main/show/2938 Description 地主某君有一块由2×n个栅格组成的土地,有k个儿子,现在地主快要终老了,要把这些土地分 ...

随机推荐

  1. 通过Java创建XML(中文乱码已解决)

    package com.zyb.xml; import java.io.FileOutputStream; import java.io.OutputStream; import java.io.Ou ...

  2. 「luogu2617」Dynamic Rankings

    「luogu2617」Dynamic Rankings 传送门 树套树直接上树状数组套主席树,常数很大就是了. 树套树参考代码: /*-------------------------------- ...

  3. 「ZJOI2007」捉迷藏

    题目描述 给出一棵\(N\)个有色(黑白,黑色对应关灯,白色对应开灯)节点的树以及\(M\)次操作,每次操作将改变一个节点的颜色或者求出树上最远的两个白点距离 基本思路 \(60pts\)做法 这道题 ...

  4. Flask - 闪现flash

    1. 像snap一样阅后即焚,在服务器端临时存储数据的地方,如显示错误信息.(也可以用session实现) 2. Flash的底层是session做的,所以要secret_key.可以看源码 3. f ...

  5. 在Windows中实现Java调用DLL(转载)

    本文提供调用本地 C 代码的 Java 代码示例,包括传递和返回某些常用的数据类型.本地方法包含在特定于平台的可执行文件中.就本文中的示例而言,本地方法包含在 Windows 32 位动态链接库 (D ...

  6. list中会直接绑定HashMap中的数据

    import java.util.ArrayList;import java.util.HashMap;import java.util.List; public class HashMapSync ...

  7. SciPy 常量

    章节 SciPy 介绍 SciPy 安装 SciPy 基础功能 SciPy 特殊函数 SciPy k均值聚类 SciPy 常量 SciPy fftpack(傅里叶变换) SciPy 积分 SciPy ...

  8. 域名配置DNS解析A记录,映射到主机

    有很多域名的供应商,随便选,哪个便宜用哪个.godaddy一直支持支付宝,不用visa,虽然它是国外的. 我用的是godaddy,这两年有中文版的了,虽然它有了中文版,但是比以前的英文版还要慢. 进入 ...

  9. Thinkcmf任意漏洞包含漏洞分析复现

    简介 ThinkCMF是一款基于PHP+MYSQL开发的中文内容管理框架,底层采用ThinkPHP3.2.3构建.ThinkCMF提出灵活的应用机制,框架自身提供基础的管理功能,而开发者可以根据自身的 ...

  10. 031.SAP上查看所有的用户账号,查询SAP用户账号的后台数据库表

    01. 输入事务代码SU11, 然后输入SAP用户账号数据表USER_ADDR 02. 点击实用程序,再点击内容 03.点击查询 04. 将查看到的结果通过Excel表格导出 不忘初心,如果您认为这篇 ...