[CF833B] The Bakery
Description
将一个长度为n的序列分为k段
使得总价值最大一段区间的价值表示为区间内不同数字的个数
\(n\leq 35000,k\leq 50,1\leq a_i\leq n\)
Solution
定义 \(dp[i][j]\) 表示前 i 个里面分 j 段的最大收益
一个显然的 dp 方程是 \(dp[i][j]=\max \limits_{1\leq p<i} dp[p][j-1]+w(p+1,i)\)。复杂度 \(O(n^2k)\),GG。
考虑优化此方程,因为是取 max,容易想到放在线段树上实现。
同时定义 \(pre[a[i]]\) 表示当前 \(a[i]\) 这个元素上一次出现的位置是哪里,如果没有出现则是 0 。
难点在于 \(w\) 数组如何动态快速的求出来,我们外层循环一个 \(j\) 表示分的段数,发现如果当前扫到 i 这个位置那么 a[i] 的贡献实际上是让 \([pre[a[i]],i]\) 这段区间整体加一。可以这么理解,就是当前扫到 i,那么对于所有到 i 截至的区间 \([p,i]\),a[i] 这个元素对这些区间有贡献的部分是左端点\(\in [pre[a[i]],i]\) 里的这一段。线段树区间加就好了。也就是说,当前扫到了 i ,那么线段树的叶子节点 p 表示的就是 \(w[p,i]\) 的值,这也是我们用线段树的意义所在。这样就可以 \(O(nlogn)\) 求出 w 数组了。同时 dp 数组实时更新即可。
还有一点要注意的是方程是 \(dp[p][j-1]+w(p+1,i)\) ,也就是说能用来更新答案的是 节点 p 的 dp 值和 p+1 的累加值,有点麻烦,干脆把所有的 dp 值都往左挪一个就行了,也就是叶子节点 p 表示的实际上是 p+1 的值。感觉有点绕。。。
Code
#include<cstdio>
#include<cctype>
#include<cstring>
#define K 55
#define N 35005
#define min(A,B) ((A)<(B)?(A):(B))
#define max(A,B) ((A)>(B)?(A):(B))
#define swap(A,B) ((A)^=(B)^=(A)^=(B))
int n,k;
int f[N];
int val[N];
int pre[N];
int mx[N<<2];
int lazy[N<<2];
int getint(){
int x=0,f=0;char ch=getchar();
while(!isdigit(ch)) f|=ch=='-',ch=getchar();
while(isdigit(ch)) x=(x<<1)+(x<<3)+(ch^48),ch=getchar();
return f?-x:x;
}
void build(int cur,int l,int r){
if(l==r){
mx[cur]=f[l-1];
return;
}
int mid=l+r>>1;
build(cur<<1,l,mid);
build(cur<<1|1,mid+1,r);
mx[cur]=max(mx[cur<<1],mx[cur<<1|1]);
}
void pushdown(int cur){
if(!lazy[cur]) return;
lazy[cur<<1]+=lazy[cur];
lazy[cur<<1|1]+=lazy[cur];
mx[cur<<1]+=lazy[cur];
mx[cur<<1|1]+=lazy[cur];
lazy[cur]=0;
}
void modify(int cur,int l,int r,int ql,int qr){
if(!ql or !qr or ql>qr) return;
if(ql<=l and r<=qr){
mx[cur]++;
lazy[cur]++;
return;
}
pushdown(cur);
int mid=l+r>>1;
if(ql<=mid)
modify(cur<<1,l,mid,ql,qr);
if(mid<qr)
modify(cur<<1|1,mid+1,r,ql,qr);
mx[cur]=max(mx[cur<<1],mx[cur<<1|1]);
}
int query(int cur,int l,int r,int ql,int qr){
if(ql<=l and r<=qr)
return mx[cur];
pushdown(cur);
int mid=l+r>>1,ans=0;
if(ql<=mid){
int p=query(cur<<1,l,mid,ql,qr);
ans=max(ans,p);
}
if(mid<qr){
int p=query(cur<<1|1,mid+1,r,ql,qr);
ans=max(ans,p);
}
return ans;
}
signed main(){
n=getint(),k=getint();
for(int i=1;i<=n;i++)
val[i]=getint();
for(int j=1;j<=k;j++){
memset(mx,0,sizeof mx);
memset(pre,0,sizeof pre);
memset(lazy,0,sizeof lazy);
build(1,1,n);
for(int i=1;i<=n;i++){
modify(1,1,n,pre[val[i]]+1,i);
pre[val[i]]=i;
//if(i<j) continue;
f[i]=query(1,1,n,1,i);
//printf("j=%d,i=%d,f=%d\n",j,i,f[i]);
}
}
printf("%d\n",f[n]);
return 0;
}
[CF833B] The Bakery的更多相关文章
- CF833B The Bakery 线段树,DP
CF833B The Bakery LG传送门 线段树优化DP. 其实这是很久以前就应该做了的一道题,由于颓废一直咕在那里,其实还是挺不错的一道题. 先考虑\(O(n^2k)\)做法:设\(f[i][ ...
- CF-833B The Bakery(线段树优化Dp)
Some time ago Slastyona the Sweetmaid decided to open her own bakery! She bought required ingredie ...
- CF833B The Bakery (线段树+DP)
题目大意:给你一个序列(n<=35000),最多分不大于m块(m<=50),求每个块内不同元素的数量之和的最大值 考试的时候第一眼建图,没建出来,第二眼贪心 ,被自己hack掉了,又随手写 ...
- DP 优化方法大杂烩 & 做题记录 I.
标 * 的是推荐阅读的部分 / 做的题目. 1. 动态 DP(DDP)算法简介 动态动态规划. 以 P4719 为例讲一讲 ddp: 1.1. 树剖解法 如果没有修改操作,那么可以设计出 DP 方案 ...
- Codeforeces 707B Bakery(BFS)
B. Bakery time limit per test 2 seconds memory limit per test 256 megabytes input standard input out ...
- Codeforces Round #368 (Div. 2) B. Bakery (模拟)
Bakery 题目链接: http://codeforces.com/contest/707/problem/B Description Masha wants to open her own bak ...
- 信号量和PV操作写出Bakery算法的同步程序
面包店烹制面包及蛋糕,由n个销售员卖出.当有顾客进店购买面包或蛋糕时,应先在取号机上取号,然后等待叫号,若有销售员空闲时便叫下一号,试用信号量和PV操作写出Bakery算法的同步程序. 设计要求 1) ...
- Codeforces 834D The Bakery【dp+线段树维护+lazy】
D. The Bakery time limit per test:2.5 seconds memory limit per test:256 megabytes input:standard inp ...
- Codeforces 834D The Bakery - 动态规划 - 线段树
Some time ago Slastyona the Sweetmaid decided to open her own bakery! She bought required ingredient ...
随机推荐
- js 定时更改div背景图片
今天遇到一个业务场景,使用js将一个div标签的背景图片定时更换一下. 之前百度了几个,有css+js,也有css3的,不过css3的兼容有问题,之后同事提示,可以使用js直接来更换div的北京图片, ...
- 达里奥:典型的去杠杆化过程是怎么进行的zz
猛人RayDalio的“三部曲”之三:关于去杠杆化的深入理解 作者系统地阐述了去杆杠化过程并深入探讨去杆杠化的运作机理,对我们理解当前全球乃至中国.即将或者已经面临的去杠杆化过程,应当能够带来一些帮助 ...
- python模块:time
# encoding: utf-8 # module time # from (built-in) # by generator 1.145 """ This modul ...
- 第四周Access的总结
一.问;这节课你学到了什么知识? 答:这周我们学习了新的课程:表的建立和管理.这是Access最重要的一部份,需要我们认认真真学习! 字段是表最重要的信息,而数据类型是字段或表中 ...
- DE1-SOC资源
1,digital solution lab 网站上的de1soc QT教程. 内容包括: Install Qt 5.4 Designer Install the Altera SoC Tool-Ch ...
- ASP.NET Core 请求/查询/响应参数格式转换(下划线命名)
业务场景: 在 ASP.NET Core 项目中,所有的代码都是骆驼命名,比如userName, UserName,但对于 WebApi 项目来说,因为业务需要,一些请求.查询和响应参数的格式需要转换 ...
- iOS逆向之Reveal
Reveal是一个强大的UI分析工具,使用它可以查看各个界面的视图层级,在解决界面显示问题时非常有用.它最大的特点就是非常直观,查看UI布局的时候非常方便. 我们知道,Reveal官网提供的方法只能监 ...
- 异步请求获取JSON数据
json格式的数据广泛应用于异步请求中前后端的数据交互,本文主要介绍几种使用场景和使用方法. <script type="text/javascript"> funct ...
- Eclipse上搭建Spring的开发环境
一.安装Spring Tool Suite插件 如图: 点击Finish之后等待安装,安装完之后弹窗点击yes重启Eclipse,重启后显示如下界面: 二.搭建Spring开发环境 1.导入jar包到 ...
- 使用config 来管理ssh的会话
通常利用 ssh 连接远程服务器,一般都要输入以下类型命令: ssh user@hostname -p port 如果拥有多个ssh账号,特别是像我这种喜欢在终端里直接ssh登录, 要记住每个ssh账 ...