洛谷 P5665 [CSP-S2019] 划分
链接:
题意:
给出 \(n\) 个整数 \(a_i\) ,你需要找到一些分界点 \(1 \leq k_1 \lt k_2 \lt \cdots \lt k_p \lt n\),使得
\(\sum\limits_{i=1}^{k_1} a_i \leq \sum\limits_{i=k_1+1}^{k_2} a_i \leq \cdots \leq \sum\limits_{i=k_p+1}^{n} a_i\)。
注意 \(p\) 可以为 \(0\) 且此时 \(k_0 = 0\)。
请你最小化
\((\sum\limits_{i=1}^{k_1} a_i)^2 + (\sum\limits_{i=k_1+1}^{k_2} a_i)^2 + \cdots + (\sum\limits_{i=k_p+1}^{n} a_i)^2\)。
分析:
根据完全平方公式有:\((a+b)^2\geq a^2+b^2\)
所以分段比不分段更优。其次,对于一个数 \(x\),将他分到左边和右边会造成 \(2x*sum_{side}+x^2\) 的贡献(\(sum\) 指两区间和),又因为 \(sum_L\leq sum_R\) 所以尽量分到左边更优。也就是说,最后一段和最小时,答案最优。(这个策略还是能猜出来,只是不敢确定)。于是就有后面的\(O(n^2)\) dp 了。
算法:
首先维护一个前缀和 \(sum\) (和上文 \(sum\) 不同)。设 \(d[i]\) 为 \(i\) 结尾,最后一段最小时上一段的结尾位置,于是有 \(d[i]=max\{j|sum[i]-sum[j]\geq sum[j]-sum[d[j]]\}\)。从 \(i\) 向左循环遇到的第一个满足条件的位置就是 \(d[i]\)。输出时从 \(n\) 不停向它的 \(d\) 值跳,一直跳到 \(0\)。复杂度 \(O(n^2)\)。
优化:
根据上述算法可以写出这样的程序:
for(int i=1;i<=n;i++){
for(int j=i-1;j>=1;j--)
if(sum[i]-sum[j]<sum[j]-sum[d[j]]) continue;
d[i]=j; break;
}
int now=n;
while(now){
int t=sum[now]-sum[d[now]];
ans+=t*t;
now=d[now];
}
\(O(n^2)\) 复杂度可以通过64分的好成绩,但是看到 \(2\leq n\leq4\times10^7\),这说明我们需要一个 \(O( n )\) 或实(hen)现(neng)良(ka)好(chang)的 \(O(n\log n)\)。
回顾算法,发现判断 \(j\) 是否合法时的柿子:
\(sum[i]-sum[j]\geq sum[j]-sum[d[j]]\)
可以继续改写:
\(2*sum[j]-sum[d[j]]\leq sum[i]\)
此时左边只与 \(j\) 有关右边只与 \(i\) 有关。设 \(A(j)=2*sum[j]-sum[d[j]]\) 。显然 \(A(j)\) 越小,\(j\) 越可能成为合法答案,所以当存在 \(j_1<j_2\) 且 \(A(j_1)>A(j_2)\) 时,\(j_2\) 比 \(j_1\) 更优。
又有 \(sum[i]\lt sum[i+1]\) 所以当一个 \(j\) 满足 \(A(j)\leq sum[i]\) 时,它也满足 \(A(j)\leq sum[i+1]\)。
基于以上两点我们可以维护出一个 \(A(j)\) 单调上升且 \(j\) 单调上升的单调队列,每次转移时找到最大的满足 \(A(j)\leq sum[i]\) 的 \(j\),小于 \(j\) 的状态可以舍弃,更新 \(d[i]\) 后将 \(A(i)\) 加入队列尾并弹出 \(A(k)>A(i)\) 的状态 \(k\),每个点最多进出1次,所以复杂度是 \(O(n)\)。
于是我们可以这样维护 \(d[i]\):
for(int i=1;i<=n;i++){
while(head<tail&&A(q[head+1])<=sum[i])head++;
d[i]=q[head];
while(head<tail&&A(i)<A(q[tail]))tail--;
q[++tail]=i;
}
另外的,此题最后三个测试点相当毒瘤,输入相当占时间和空间,数据范围会爆 long long。我们不得不选择高精(或考场上不敢写的__int128),同时需要对空间和时间能够精确把控,最好还是自己慢慢调,可以锻炼自己的代码能力。
代码:
#include<bits/stdc++.h>
using namespace std;
#define ll long long
#define in read()
inline int read(){
int p=0,f=1;
char c=getchar();
while(c>'9'||c<'0'){if(c=='-')f=-1;c=getchar();}
while(c>='0'&&c<='9'){p=p*10+c-'0';c=getchar();}
return p*f;
}
const int N=4e7+5;
ll sum[N];
int d[N],q[N],n,type,head,tail;
inline __int128 A(int i){return 2*sum[i]-sum[d[i]];}
//sub23~25
const int M=1e5+5;
const int mod=1ll<<30;
ll x,y,z,m;
int p[M],l[M],r[M];
ll b[N];
//
__int128 ans;
void print(__int128 x){
if(x==0){
cout<<0;
return ;
}
string res="";
while(x){
res+=x%10+'0';
x/=10;
}
reverse(res.begin(),res.end());
cout<<res;
}
signed main(){
n=in,type=in;
if(type==0)
for(int i=1;i<=n;i++)
sum[i]=sum[i-1]+in;
else{
x=in,y=in,z=in,b[1]=in,b[2]=in,m=in;
for(int i=1;i<=m;i++)
p[i]=in,l[i]=in,r[i]=in;
for(int i=3;i<=n;i++)
b[i]=((x*b[i-1]%mod+y*b[i-2]%mod)%mod+z)%mod;
int now=0;
for(int i=1;i<=n;i++){
if(i>p[now])now++;
sum[i]=sum[i-1]+b[i]%(r[now]-l[now]+1)+l[now];
}
}
for(int i=1;i<=n;i++){
while(head<tail&&A(q[head+1])<=sum[i])head++;
d[i]=q[head];
while(head<tail&&A(i)<A(q[tail]))tail--;
q[++tail]=i;
}
int now=n;
while(now){
__int128 t=sum[now]-sum[d[now]];
ans+=t*t;
now=d[now];
}
print(ans);
return 0;
}
可能是因其毒瘤的数据才成为了紫题
洛谷 P5665 [CSP-S2019] 划分的更多相关文章
- 洛谷P4047 [JSOI2010]部落划分题解
洛谷P4047 [JSOI2010]部落划分题解 题目描述 聪聪研究发现,荒岛野人总是过着群居的生活,但是,并不是整个荒岛上的所有野人都属于同一个部落,野人们总是拉帮结派形成属于自己的部落,不同的部落 ...
- uoj#348/洛谷P4221 [WC2018]州区划分(FWT)
传送门(uoj) 传送门(洛谷) 全世界都会子集卷积就咱不会--全世界都在写\(FMT\)就咱只会\(FWT\)-- 前置芝士 或运算\(FWT\)或者\(FMT\) 左转洛谷模板区,包教包会 子集卷 ...
- 【CSP-S 2019】【洛谷P5665】划分【单调队列dp】
前言 \(csp\)时发现自己做过类似这道题的题目 : P4954 [USACO09Open] Tower of Hay 干草塔 然后回忆了差不多\(15min\)才想出来... 然后就敲了\(88p ...
- 洛谷——P1025 数的划分
P1025 数的划分 题目描述 将整数n分成k份,且每份不能为空,任意两个方案不相同(不考虑顺序). 例如:n=7,k=3,下面三种分法被认为是相同的. 1,1,5; 1,5,1; 5,1,1; 问有 ...
- 洛谷 P1025 数的划分 Label:dp
题目描述 将整数n分成k份,且每份不能为空,任意两个方案不相同(不考虑顺序). 例如:n=7,k=3,下面三种分法被认为是相同的. 1,1,5; 1,5,1; 5,1,1; 问有多少种不同的分法. 输 ...
- 洛谷 P1025 数的划分
题目描述 将整数n分成k份,且每份不能为空,任意两个方案不相同(不考虑顺序). 例如:n=7,k=3,下面三种分法被认为是相同的. 1,1,5; 1,5,1; 5,1,1; 问有多少种不同的分法. 输 ...
- [NOIP2001] 提高组 洛谷P1025 数的划分
题目描述 将整数n分成k份,且每份不能为空,任意两个方案不相同(不考虑顺序). 例如:n=7,k=3,下面三种分法被认为是相同的. 1,1,5; 1,5,1; 5,1,1; 问有多少种不同的分法. 输 ...
- 洛谷P1025 数的划分【dp】
将整数nn分成kk份,且每份不能为空,任意两个方案不相同(不考虑顺序). 例如:n=7n=7,k=3k=3,下面三种分法被认为是相同的. 1,1,51,1,5; 1,5,11,5,1; 5,1,15, ...
- nbuoj 2080 洛谷p1025 数的划分
链接:http://www.nbuoj.com/v8.83/Problems/Problem.php?pid=2820 链接:https://www.luogu.org/problem/P1025 题 ...
随机推荐
- .NET 中的HTTP 3支持
dotnet团队官方博客发布了一篇HTTP3的文章:HTTP/3 support in .NET 6.文章介绍了.NET 6 将预览支持HTTP3,.NET 7正式支持HTTP3,原因主要是HTTP/ ...
- Docker入门系列之二:Docker术语
原文作者:Jeff Hale 原文地址:https://towardsdatascience.com/learn-enough-docker-to-be-useful-1c40ea269fa8 翻译: ...
- PHP中的输出:echo、print、printf、sprintf、print_r和var_dump
大家在面试中,经常会被问到的问题: 请简要说明PHP的打印方式都有哪些? 或者直接点问: 请说明echo.print.print_r的区别 看着很简单,一般会出现在初中级的笔试题中.但是要真正说明白这 ...
- python继承细节
不要子类化内置类型 内置类型(由C语言编写)不会调用用户定义的类覆盖的特殊方法. 例如,子类化dict作为测验: class DoppeDict(dict): def __setitem__(self ...
- 鸿蒙内核源码分析(内存规则篇) | 内存管理到底在管什么 | 百篇博客分析OpenHarmony源码 | v16.02
百篇博客系列篇.本篇为: v16.xx 鸿蒙内核源码分析(内存规则篇) | 内存管理到底在管什么 | 51.c.h .o 内存管理相关篇为: v11.xx 鸿蒙内核源码分析(内存分配篇) | 内存有哪 ...
- P3343-[ZJOI2015]地震后的幻想乡【dp,数学期望】
正题 题目链接:https://www.luogu.com.cn/problem/P3343 题目大意 给出\(n\)个点的一张无向图,每条边被修复的时间是\([0,1]\)的一个随机实数,求这张图联 ...
- P3273-[SCOI2011]棘手的操作【线段树,并查集】
正题 题目链接:https://www.luogu.com.cn/problem/P3273 题目大意 \(n\)个点有权值,要求支持操作 连接两个点 单点加权 联通块加权 全图加权 单点询问 联通块 ...
- P5137-polynomial【倍增】
正题 题目链接:https://www.luogu.com.cn/problem/P5137 题目大意 \(T\)组数据给出\(n,a,b,p\)求 \[\left(\sum_{0=1}^na^ib^ ...
- Stream聚合函数
Stream班介绍 幼稚园开学的第一天,各们家长把小朋友送到了园里,各位小朋友都你看看我,我看看你.有的嚎啕大哭,有的呆若木鸡....这里时候园长安排我拿来小本本记录入园的小朋友.... 记录小朋友 ...
- Jenkins持续集成与部署
一.Jenkins简介 在阅读此文章之前,你需要对Linux.Docker.Git有一定的了解和使用,如果还未学习,请阅读我前面发布的相关文章进行学习. 1.概念了解:CI/CD模型 CI全名Cont ...