_{_{(宋体字看起来真舒服)}}

(宋体字看起来真舒服)​​

(

)

题面_{_{(洛谷翻译)}}

题面(洛谷翻译)​​

给定

n

n

n个不降的数组。
有一个值

a

n

s

ans

ans,初始为

0

0

0
你需要进行如下操作

k

k

k次:

1.选择一个数组
2.把

a

n

s

ans

ans加上数组的第一个元素。
3.把数组的第一个元素删除。

请求出

a

n

s

ans

ans最大是多少。

所有数组的元素总个数

1

0

6

,

n

,

k

3000

\leq 10^6,n,k\leq 3000

≤106,n,k≤3000

(

)

输入_{_{(笔者亲自翻译)}}

输入(笔者亲自翻译)​​

第一行包含两个整数

n

n

n and

k

k

k (

1

n

,

k

3

000

1 \le n, k \le 3\,000

1≤n,k≤3000 ): 分别是数组和操作的数量。

接下来的

n

n

n 行每一行包含一个数组。 每一行的第一个整数是

t

i

t_i

ti​ (

1

t

i

1

0

6

1 \le t_i \le 10^6

1≤ti​≤106 ): 第

i

i

i 个数组的大小。 后面

t

i

t_i

ti​ 个整数

a

i

,

j

a_{i, j}

ai,j​ (

0

a

i

,

1

a

i

,

t

i

1

0

8

0 \le a_{i, 1} \le \ldots \le a_{i, t_i} \le 10^8

0≤ai,1​≤…≤ai,ti​​≤108 ) 给出了第

i

i

i 个数组。

保证

k

i

=

1

n

t

i

1

0

6

k \le \sum\limits_{i=1}^n t_i \le 10^6

k≤i=1∑n​ti​≤106。

样例

样例

i

n

p

u

t

input

input

3 3
2 5 10
3 1 2 3
2 1 20
o

u

t

p

u

t

output

output

26

题解

题解

转换一下模型,相当于一个背包问题,这个不用我多说了吧。

我们会发现一个结论:最后的最优操作一定是把一些数组全选完,最多一个数组只选部分。

S

t

e

p
  

1
  

Step\;1\;证明结论

Step1证明结论

我们要证这个结论,实际上要证最优操作不存在两个只选了部分的数组

为了方便证明,我们定义两个数组总和相等的时候,两数组选的数量相差更大的更优

其实很简单,如果存在这么两个数组选了部分:

那么以下两种情况肯定更优:

  1. a

    i

    ,

    x

    a

    j

    ,

    y

    a

    i

    ,

    x

    +

    1

    a

    j

    ,

    y

    :

    a_{i,x} ≥ a_{j,y} \Rightarrow a_{i,x+1} ≥ a_{j,y}:

    ai,x​≥aj,y​⇒ai,x+1​≥aj,y​:

  2. a

    i

    ,

    x

    a

    j

    ,

    y

    a

    j

    ,

    y

    +

    1

    a

    i

    ,

    x

    :

    a_{i,x} ≤ a_{j,y} \Rightarrow a_{j,y+1} ≥ a_{i,x}:

    ai,x​≤aj,y​⇒aj,y+1​≥ai,x​:

而这两种情况已经包含全集了(

a

i

,

x

a

j

,

y

a_{i,x} ≥ a_{j,y}

ai,x​≥aj,y​ 和

a

i

,

x

a

j

,

y

a_{i,x} ≤ a_{j,y}

ai,x​≤aj,y​),所以充分证明了 两个数组都没选完,在中间稳定 的情况是不存在的,因此最多就只有其中一个选完了,另一个没选完的情况。

S

t

e

p
  

2
  

Step\;2\;利用结论

Step2利用结论

这个结论怎么利用呢?

相当于我们如果确定某一个数组是只选部分的那一个,那么其他的数组就降维了、变成一个数了、一个物品了!

于是我们可以先把其他数组假设成只有一个数,然后把背包跑出来,最后暴力枚举只选部分的那个数组具体选几个。

那么,每一个数组一开始都有一个空的背包dp数组,表示该数组只选部分的情况下其他数组放进来处理的背包。但是这样还是

O

(

n

2

k

)

O(n^2k)

O(n2k) 的过不了。

我们可以想想怎么优化。我们发现每个数组

i

i

i 整个考虑会对

[

1

,

i

1

]

[

i

+

1

,

n

]

[1,i-1]∪[i+1,n]

[1,i−1]∪[i+1,n] 内的其他数组的背包有贡献,这是两个区间,因此我们可以用线段树分治优化它,

这样一来每个数组就要对

l

o

g

n

logn

logn 个区间有贡献,而每次贡献是一次

O

(

m

)

O(m)

O(m) 的背包更新,时间复杂度

O

(

n

m

l

o

g

n

)

O(nmlogn)

O(nmlogn)。

最后,由于每个数组挂的区间有特殊性,可以不用真的建一棵线段树,直接判断挂在当前区间的数组有哪些就行了。

C

O

D

E

CODE

CODE

#include<queue>
#include<cmath>
#include<vector>
#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
using namespace std;
#define MAXN 3005
#define LL long long
#define DB double
#define ENDL putchar('\n')//#define lowbit(x) ((-x)&(x))
LL read() {
LL f = 1,x = 0;char s = getchar();
while(s < '0' || s > '9') {if(s=='-')f = -f;s = getchar();}
while(s >= '0' && s <= '9') {x=x*10+(s-'0');s = getchar();}
return f * x;
}
const int MOD = 998244353;
int n,m,i,j,s,o,k,sq;
LL dp[100][MAXN];
bool f[100][MAXN];
vector<LL> a[MAXN];
int b[MAXN];
LL ans = 0;
void solve(int l,int r,int de) {
memcpy(dp[de],dp[de-1],sizeof(dp[de]));
memcpy(f[de],f[de-1],sizeof(f[de]));
for(int i = 1;i <= n;i ++) {
if(i == l) {
i = r;continue;
}
if(!f[de][i]) {
f[de][i] = 1;
int tp = a[i][0];
for(int j = m;j >= tp;j --) {
dp[de][j] = max(dp[de][j],dp[de][j-tp] + a[i][tp]);
}
}
}
if(l < r) {
int mid = (l + r) >> 1;
solve(l,mid,de+1);solve(mid+1,r,de+1);
return ;
}
else {
int tp = a[l][0];
ans = max(ans,dp[de][m]);
for(int i = 1;i <= tp && i <= m;i ++) {
ans = max(ans,dp[de][m-i] + a[l][i]);
}
}
return ;
}
int main() {
n = read();m = read();
for(int i = 1;i <= n;i ++) {
s = read();
LL sm = 0;
a[i].push_back(s);
while(s --) {
sm += read();
a[i].push_back(sm);
}
}
solve(1,n,1);
printf("%lld\n",ans);
return 0;
}

CF1442D Sum (动态规划,线段树分治)的更多相关文章

  1. 【洛谷P4319】 变化的道路 线段树分治+LCT

    最近学了一下线段树分治,感觉还蛮好用... 如果正常动态维护最大生成树的话用 LCT 就行,但是这里还有时间这一维的限制. 所以,我们就把每条边放到以时间为轴的线段树的节点上,然后写一个可撤销 LCT ...

  2. LOJ2312 LUOGU-P3733「HAOI2017」八纵八横 (异或线性基、生成树、线段树分治)

    八纵八横 题目描述 Anihc国有n个城市,这n个城市从1~n编号,1号城市为首都.城市间初始时有m条高速公路,每条高速公路都有一个非负整数的经济影响因子,每条高速公路的两端都是城市(可能两端是同一个 ...

  3. BZOJ_1672_[Usaco2005 Dec]Cleaning Shifts 清理牛棚_动态规划+线段树

    BZOJ_1672_[Usaco2005 Dec]Cleaning Shifts 清理牛棚_动态规划+线段树 题意:  约翰的奶牛们从小娇生惯养,她们无法容忍牛棚里的任何脏东西.约翰发现,如果要使这群 ...

  4. loj#2312. 「HAOI2017」八纵八横(线性基 线段树分治)

    题意 题目链接 Sol 线性基+线段树分治板子题.. 调起来有点自闭.. #include<bits/stdc++.h> #define fi first #define se secon ...

  5. BZOJ.4184.shallot(线段树分治 线性基)

    BZOJ 裸的线段树分治+线性基,就是跑的巨慢_(:з」∠)_ . 不知道他们都写的什么=-= //41652kb 11920ms #include <map> #include < ...

  6. BZOJ.4137.[FJOI2015]火星商店问题(线段树分治 可持久化Trie)

    BZOJ 洛谷 一直觉得自己非常zz呢.现在看来是真的=-= 注意题意描述有点问题,可以看BZOJ/洛谷讨论. 每个询问有两个限制区间,一是时间限制\([t-d+1,t]\),二是物品限制\([L,R ...

  7. 洛谷.3733.[HAOI2017]八纵八横(线性基 线段树分治 bitset)

    LOJ 洛谷 最基本的思路同BZOJ2115 Xor,将图中所有环的异或和插入线性基,求一下线性基中数的异或最大值. 用bitset优化一下,暴力的复杂度是\(O(\frac{qmL^2}{w})\) ...

  8. bzoj4025二分图(线段树分治 并查集)

    /* 思维难度几乎没有, 就是线段树分治check二分图 判断是否为二分图可以通过维护lct看看是否链接出奇环 然后发现不用lct, 并查集维护奇偶性即可 但是复杂度明明一样哈 */ #include ...

  9. BZOJ3237:[AHOI2013]连通图(线段树分治,并查集)

    Description Input Output Sample Input 4 5 1 2 2 3 3 4 4 1 2 4 3 1 5 2 2 3 2 1 2 Sample Output Connec ...

随机推荐

  1. 【转载】k8s入坑之路(2)kubernetes架构详解

    每个微服务通过 Docker 进行发布,随着业务的发展,系统中遍布着各种各样的容器.于是,容器的资源调度,部署运行,扩容缩容就是我们要面临的问题. 基于 Kubernetes 作为容器集群的管理平台被 ...

  2. 【由浅入深_打牢基础】HOST头攻击

    [由浅入深_打牢基础]HOST头攻击 前几天一直准备别的事情,然后用了2/3天时间去挖了补天某厂的SRC,还是太菜了,最后提交了一个低危(还没出结果,还有点敏感信息泄露,感觉略鸡肋也没交),不过偶然发 ...

  3. MySQL-4-DDL

    DDL:数据定义语言 创建create 创建库 语法:create database [if not exists]库名 # 创建库 CREATE DATABASE IF NOT EXISTS boo ...

  4. CH341驱动安装

    CH341驱动安装 参考文章:https://blog.csdn.net/qq_33194301/article/details/104510078 方法一: 下载驱动包,按提示编译,会出现下面报错 ...

  5. Eclipse Ctrl+鼠标左键不能查看源代码

    查询当前项目的使用的java包版本. 找到java包相应版本的安装路径. 在 "Source Attachment"对话框下,选择"External location&q ...

  6. VBA驱动SAP GUI完成界面元素值初始化

    小爬日常利用VBA完成SAP GUI自动化时,经常被这个问题困扰:我们进入一个事务代码界面时,如FBL1N(供应商行项目显示),很多的 GuiTextField(文本框)对象.GuiCheckBox( ...

  7. 使用Java客户端发送消息和消费的应用

    体验链接:https://developer.aliyun.com/adc/scenario/fb1b72ee956a4068a95228066c3a40d6 实验简介 本教程将Demo演示使用jav ...

  8. 博客从 CloudBase 迁移至云主机

    迁移起因 原来的博客 其实从很久以前就想要写博客,但总是断断续续的,一直都没有认真地开始. 最终,决定使用静态博客工具作为自己博客的承载体.在多种工具的比较下,最终选择了 Hugo 并部署到 Gith ...

  9. c++头文件的一个误导

    通常情况下,我们都认为c++的头文件是这样的: #include <bits/stdc++.h> using namespace std; int main() 但c++也可以用c的头文件 ...

  10. CF1702A Round Down the Price 题解

    题意:给定一个数 \(n\),找出一个数为 \(10^k \leq n\),求二者的差. 建立一个数组,储存 \(10^k\),每次直接查询求差输出. 注意数据范围. #include<cstd ...