可以将问题用形象的方式来表述。给定一排点,第 \(i\) 个点有它需要的覆盖次数 \(a_i\)。有两种线段,一种能覆盖连续的一些点,称其为连续线段;另一种能覆盖相邻间隔为 \(1\) 的一些点,称其为为间隔线段。现在要用尽可能少的线段覆盖每个点 \(i\) 恰好 \(a_i\) 次。

发现如果没有间隔线段就是被某组织出烂的原题。显然,每次取最长的非 \(0\) 段最优。

回到本题,来证明一个结论:假设 \(a_1\sim a_i>0\),\(a_{i+1}=0\) 且 \(i>1\),选一条从 \(1\sim i\) 的连续线段一定不劣。

将 \(1\sim i\) 分成奇数偶数两组,那么每组都可以用间隔线段完全覆盖且不会影响到另一组,这意味着如果不用连续线段两者可以用上面的方法单独计算

反之如果我们选择了一条 \(1\sim i\) 的连续线段,两组均会被完全覆盖,代价减少了 \(1\),看起来会更优

但现在点 \(i\) 所在的组可能对后面产生贡献(覆盖到点 \(i+2\)),而另一组不行。所以我们可以以点 \(i+2\) 为开头选一条间隔线段,加上 \(1\sim i\) 的连续线段,代价为 \(2\),不劣于选两条间隔线段。

接着怎么做呢?唯一需要考虑的就是前面有线段连过来。在处理点 \(i\) 时,记连过来的连续线段数量为 \(x\),能覆盖到点 \(i\) 的间隔线段数量为 \(y\),不能覆盖到点 \(i\) 的间隔线段数量为 \(z\)。

首先处理一下特殊情况:连过来的线段太多了,即 \(x+y>a_i\),有 \(k\gets x+y-a_i\) 条线段不能向后继续连。但我们会发现留下间隔线段还是连续线段是由后面的点决定的,所以不妨将 \(x\gets x-k\),\(y\gets y-k\),反悔标记置为 \(k\),表示可以免费连向下一个点的线段个数(线段的种类可以任意选)。但是这样会出现一种情况,\(x<k\) 或 \(y<k\),此时的反悔可能是不合法的。对于这种情况,我们将少的一种线段置为 \(0\),另一种线段对应减少,即可保证合法。

然后就好办了,根据之前的结论,处理 \(a_{i-1}\) 与 \(a_i\)。能用连续线段就用,如果 \(i-1\) 还有剩下就用间隔线段。

当然还有一些需要注意的细节。

  • 如果更新了标记,最后需要将 \(a_i\) 置为 \(k\),答案减去 \(k\),同时清空标记。
  • 尽量从 \(i=2\) 开始做,防止出现一些边界问题。
  • 最后多做一次,计算 \(n\) 位置的答案。

时间复杂度 \(O\left(n\right)\)。

code:

#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
#define Min(x,y)((x)<(y)?x:y)
int read()
{
int A;
bool K;
char C;
C=A=K=0;
while(C<'0'||C>'9')K|=C=='-',C=getchar();
while(C>'/'&&C<':')A=(A<<3)+(A<<1)+(C^48),C=getchar();
return(K?-A:A);
}
int main()
{
int t,n;
ll last,now,k,x,y,z,tag,ans,tmp;
t=read();
while(t--)
{
n=read(),last=read();
ans=tag=x=y=z=0;
while(n--)
{
now=(n?read():0);
if(now<x+y)
//处理多余的线段
{
k=x+y-now;
//必须要结尾的线段
if(x<k)y-=k-x,k=x;
if(y<k)x-=k-y,k=y;
x-=k;
y-=k;
now-=k;
tag=k;
//向后所能免费延伸的线段
}
now-=x+y;
//新的必须经过当前点的线段
tmp=Min(last,now);
ans+=tmp;
last-=tmp;
now-=tmp;
x+=tmp;
//用连续线段覆盖
ans+=last;
z+=last;
//用间隔线段覆盖
last=now|tag;
ans-=tag;
tag=0;
//处理标记
swap(y,z);
//奇偶互换
}
cout<<ans<<endl;
}
return 0;
}

P6631 [ZJOI2020] 序列的更多相关文章

  1. Solution -「ZJOI 2020」「洛谷 P6631」序列

    \(\mathcal{Description}\)   Link.   给定一个长为 \(n\) 的非负整数序列 \(\lang a_n\rang\),你可以进行如下操作: 取 \([l,r]\),将 ...

  2. 【夯实PHP基础】UML序列图总结

    原文地址 序列图主要用于展示对象之间交互的顺序. 序列图将交互关系表示为一个二维图.纵向是时间轴,时间沿竖线向下延伸.横向轴代表了在协作中各独立对象的类元角色.类元角色用生命线表示.当对象存在时,角色 ...

  3. Windows10-UWP中设备序列显示不同XAML的三种方式[3]

    阅读目录: 概述 DeviceFamily-Type文件夹 DeviceFamily-Type扩展 InitializeComponent重载 结论 概述 Windows10-UWP(Universa ...

  4. 软件工程里的UML序列图的概念和总结

    俗话说,自己写的代码,6个月后也是别人的代码……复习!复习!复习! 软件工程的一般开发过程:愿景分析.业务建模,需求分析,健壮性设计,关键设计,最终设计,实现…… 时序图也叫序列图(交互图),属于软件 ...

  5. python序列,字典备忘

    初识python备忘: 序列:列表,字符串,元组len(d),d[id],del d[id],data in d函数:cmp(x,y),len(seq),list(seq)根据字符串创建列表,max( ...

  6. BZOJ 1251: 序列终结者 [splay]

    1251: 序列终结者 Time Limit: 20 Sec  Memory Limit: 162 MBSubmit: 3778  Solved: 1583[Submit][Status][Discu ...

  7. 最长不下降序列nlogn算法

    显然n方算法在比赛中是没有什么用的(不会这么容易就过的),所以nlogn的算法尤为重要. 分析: 开2个数组,一个a记原数,f[k]表示长度为f的不下降子序列末尾元素的最小值,tot表示当前已知的最长 ...

  8. [LeetCode] Sequence Reconstruction 序列重建

    Check whether the original sequence org can be uniquely reconstructed from the sequences in seqs. Th ...

  9. [LeetCode] Binary Tree Longest Consecutive Sequence 二叉树最长连续序列

    Given a binary tree, find the length of the longest consecutive sequence path. The path refers to an ...

随机推荐

  1. git的远程分支是干啥的,和本地的有什么区别?

    不知道大家有没有经历过,当我们切换到了一个新的分支想要提交代码的时候,总会遇到这样的错误. 我们把日志里的英文翻译过来是说,我们当前的分支没有设置任何上游分支.然后git提示我们可以运行下面这行代码来 ...

  2. Spring入门-------------1

    Spring是一个开源框架,为简化企业级应用开发而生,使用Spring可以使简单的JavaBean 实现以前只有EJB才能实现的功能.Spring 是一个IOC和Aop容器框架. 特性: 轻量级:Sp ...

  3. Flask中的MTV架构之Templates

    Flask 中的MTV架构之Templates 关注公众号"轻松学编程"了解更多. 1.Templates(模板引擎) 1.1 说明 ​ 模板文件就是按照特定规则书写的一个负责展示 ...

  4. Elasticsearch原理解析与性能调优

    基本概念 定义 一个分布式的实时文档存储,每个字段 可以被索引与搜索 一个分布式实时分析搜索引擎 能胜任上百个服务节点的扩展,并支持 PB 级别的结构化或者非结构化数据 用途 全文检索 结构化搜索 分 ...

  5. 《为研发同学定制的MySQL面试指南》-- 连载中

    Hi大家好,我是来自博客园的赐我白日梦! 为大家带来MySQL面试专题!全文110篇!以问答的方式,由浅入深的帮你应对各类MySQL面试题的狂轰滥炸!当然也不乏会分享一些高阶读写分离数据库中间件原理及 ...

  6. 痞子衡嵌入式:超级下载算法(RT-UFL)开发笔记(1) - 执行在不同CM内核下

    大家好,我是痞子衡,是正经搞技术的痞子.今天痞子衡给大家介绍的是超级下载算法开发笔记(1)之执行在不同CM内核下. 文接上篇 <RT-UFL - 一个适用全平台i.MXRT的超级下载算法设计&g ...

  7. Spider_基础总结2_Requests异常

    # 1: BeautifulSoup的基本使用: import requests from bs4 import BeautifulSoup html=requests.get('https://ww ...

  8. nginx&http 第三章 ngx 事件event accept epoll /init

    tcp 三次握手成功后,listen fd  可读,在process_event_timer 中调用rev->handler(rev)处理: 其回调函数为: ngx_event_accept / ...

  9. netfilter内核态与用户态 通信 之 sockopt

    用户态与内核态交互通信的方法不止一种,sockopt是比较方便的一个,写法也简单.缺点就是使用 copy_from_user()/copy_to_user()完成内核和用户的通信, 效率其实不高, 多 ...

  10. ceph使用memdisk做journal

    记得在很久很久以前,ceph当时的版本是有提供使用内存做journal的配置的,当时是使用的tmpfs,但是现在的版本在搜资料的时候,发现关于这个的没怎么找到资料,邮件列表里面有人有提到怎么做,看了下 ...