我们先考虑一下拿部分分:

subtask1

考虑因为 \(n < 2k\) ,那么我们的划分一定是从中间某个地方裁开,且满足 \(k\) 的条件的,我们发现当划分点在 \([n\ mod\ k,k]\)时满足条件,那么我们只需要维护一个前缀最大值和后缀最大值就好了。

		for(int i = 1;i <= n;++i)
pre[i] = std::max(pre[i - 1],a[i]);
for(int i = n;i >= 1;--i)
s[i] = std::max(s[i + 1],a[i]);
ll ans = 0;
for(int i = n % k ;i <= k;++i)
ans = std::max(ans,1ll * pre[i] + s[i + 1]);
std::cout<<ans<<std::endl;

subtask2

我们发现这一档的分的关键是 \(k\) 很小,那么我们先轻松就能想到一个朴素的 \(dp\) ,我们设 \(f_i\) 为以 \(i\) 处为结尾的划分的答案最大值。为了满足转移次数最小,我们每次转移都要附上一个较大的代价 \(INF\) 。

所以转移方程为 \(f_i = max_{j = max(0,i - k + 1)}(f_j + p(j + 1,i)) - INF,p(x,y) = max_{i = x} ^ y a_i\) 。

最后我们发现转移次数我们是可以计算出来的,我们为 \(f_n\) 加上 \(\lfloor\frac{n}{k}\rfloor + [n \ mod\ k > 0] * INF\) 就好了。

	for(int i = 1;i <= n;++i)
f[i] = -9e18;
// std::cout<<f[0]<<std::endl;
for(int i = 1;i <= n;++i){
ll ma = a[i];
ll m = std::max((ll)0,i - k + 1);
f[0] = 0;
for(int j = i;j >= m;--j){
ma = std::max(a[j],ma);
f[i] = std::max(1ll * f[j - 1] + ma - INF,f[i]);
}
// std::cout<<f[i] + (i / k + (i % k > 0)) * INF<<std::endl;
}
std::cout<<1ll * f[n] + 1ll * (n / k + (n % k > 0)) * INF<<std::endl;

all subtask

我们考虑 \(p(x,y)\) 是一个较难处理的点,我们可以使用单调栈来处理他。

即我们依次向右扩展当前节点,并维护最大值的阶段。

我们需要一个可以处理区间修改,区间查询最大值的数据结构,线段树。

但是要注意一个最大值区间所对应的贡献区间要向左移一位。

// Problem: P6647 [CCC 2019] Tourism
// Contest: Luogu
// URL: https://www.luogu.com.cn/problem/P6647
// Memory Limit: 128 MB
// Time Limit: 4000 ms
//
// Powered by CP Editor (https://cpeditor.org) #include<iostream>
#include<cstdio>
#include<cstring>
#define ll long long
#define N 1000005 const ll INF = 1e12;
ll n,k;
int a[N];
ll f[N]; struct seg{int l,r;ll v,tag;}t[N * 6]; ll st[N],tp; #define ls(x) (x << 1)
#define rs(x) (x << 1 | 1)
#define mid ((l + r) >> 1) inline void build(int u,int l,int r){
t[u].l = l;
t[u].r = r;
if(l == r)
return ;
build(ls(u),l,mid);
build(rs(u),mid + 1,r);
} inline void up(int u){
t[u].v = std::max(t[ls(u)].v,t[rs(u)].v);
} inline void push_down(int u){
if(t[u].tag){
t[ls(u)].tag += t[u].tag;
t[ls(u)].v += t[u].tag;
t[rs(u)].tag += t[u].tag;
t[rs(u)].v += t[u].tag;
t[u].tag = 0;
}
} inline void md(int u,int tl,int tr,ll p){
// std::cout<<u<<" "<<t[u].l<<" "<<t[u].r<<" "<<tl<<" "<<tr<<" "<<p<<std::endl;
ll l = t[u].l,r = t[u].r;
if(tl <= l && r <= tr){
t[u].v += p;
t[u].tag += p;
return ;
}
push_down(u);
if(tl <= mid)
md(ls(u),tl,tr,p);
if(tr > mid)
md(rs(u),tl,tr,p);
up(u);
} inline ll find(int u,int tl,int tr){
ll ans = -9e18;
push_down(u);
int l = t[u].l,r = t[u].r;
if(tl <= l && r <= tr)
return t[u].v;
if(tl <= mid)
ans = std::max(find(ls(u),tl,tr),ans);
if(tr > mid)
ans = std::max(find(rs(u),tl,tr),ans);
return ans;
} int main(){
scanf("%lld%lld",&n,&k);
for(int i = 1;i <= n;++i)
scanf("%lld",&a[i]);
build(1,0,n);
for(int i = 1;i <= n;++i){
while(tp && a[st[tp]] <= a[i]){
if(a[st[tp]] == a[i]){--tp;continue;}
md(1,st[tp - 1],st[tp] - 1,a[i] - a[st[tp]]);//[st[tp] + 1,st[tp + 1]]向左移一位
--tp;
}
st[++tp] = i;
md(1,i - 1,i - 1,f[i - 1] + a[i]);
ll ans = find(1,std::max(i - k,(ll)0),i - 1);
f[i] = ans - 1ll * INF;
}
// for(int i = 1;i <= n;++i)
// std::cout<<f[i] + (i - 1 + k) / k * INF<<std::endl;
std::cout<<f[n] + (n - 1 + k) / k * INF; }

[CCC​2019] Tourism题解的更多相关文章

  1. EOJ Monthly 2019.2 题解(B、D、F)

    EOJ Monthly 2019.2 题解(B.D.F) 官方题解:https://acm.ecnu.edu.cn/blog/entry/320/ B. 解题 单测试点时限: 2.0 秒 内存限制:  ...

  2. 「ZJOI2019」&「十二省联考 2019」题解索引

    「ZJOI2019」&「十二省联考 2019」题解索引 「ZJOI2019」 「ZJOI2019」线段树 「ZJOI2019」Minimax 搜索 「十二省联考 2019」 「十二省联考 20 ...

  3. AtCoder ExaWizards 2019 简要题解

    AtCoder ExaWizards 2019 简要题解 Tags:题解 link:https://atcoder.jp/contests/exawizards2019 很水的一场ARC啊,随随便便就 ...

  4. [题解][Codeforces]Good Bye 2019 简要题解

    构造题好评,虽然这把崩了 原题解 A 题意 二人游戏,一个人有 \(k_1\) 张牌,另一个人 \(k_2\) 张,满足 \(2\le k_1+k_2=n\le 100\),每张牌上有一个数,保证所有 ...

  5. Atcoder Yahoo Programming Contest 2019 简要题解

    A-C 直接放代码吧. A int n,k; int main() { n=read();k=read(); puts(k<=(n+1)/2?"YES":"NO&q ...

  6. CSP-S 2019 简要题解

    从这里开始 又考炸了,sad.....明年应该在准备高考了,考完把坑填了好了. 一半题都被卡常,qswl.[我汤姆要报警.jpg] dfs 怎么这么慢呀,sad..... i7 牛逼! 写的比较混乱, ...

  7. CF round 623 Div.1D Tourism 题解

    题目链接:https://codeforces.com/contest/1314/problem/D 大意: \(n\) 个顶点的有向图,顶点编号为 \(1\) 到 \(n\),任意两个不同的顶点 \ ...

  8. P5290 [十二省联考2019]春节十二响

    题目地址:P5290 [十二省联考2019]春节十二响 骗分方法 如果你实在一点思路也没有,暴力都不会打,那么请考虑一下骗分. 方法一 输出所有 \(M\) 的和. 期望得分:0分. 实际还有5分 方 ...

  9. HNOI2019 简要题解

    HNOI 2019 简要题解 没想到自己竟也能有机会写下这篇题解呢. LOJ Luogu Day1T1 鱼 枚举\(AD\)两点后发现\(BC\)与\(EF\)相对独立,因此只需要计算合法的\(BC\ ...

随机推荐

  1. 华为在HDC2021发布全新HMS Core 6 宣布跨OS能力开放

    [2021年10月22日·东莞]华为开发者大会 2021(Together)于今天正式开幕,华为在主题演讲中正式发布全新的HMS Core 6,向全球开发者开放7大领域的69个Kit和21,738个A ...

  2. Go语言核心36讲(Go语言进阶技术六)--学习笔记

    12 | 使用函数的正确姿势 在前几期文章中,我们分了几次,把 Go 语言自身提供的,所有集合类的数据类型都讲了一遍,额外还讲了标准库的container包中的几个类型. 在几乎所有主流的编程语言中, ...

  3. vue3.x相对于vue2.x生命周期改动

    vue3.x已经正式发布了,部分小伙伴已经用了vue3.x开发,部分小伙伴还在观望中,下面是两个影响比较大的改动 1.beforeDestroy和destroyed不能用了. 这个应该是vue2.x项 ...

  4. 21.7.31 test

    \(NOIP\) 测试 好久没有这种感觉能阿克的冲动了!但还是挂了分 T1 WOJ2608(模拟,拓扑排序) 签到题,直接模拟,有点像拓扑排序. 要给点打标记不然可能被某次操作中弹出多次该点导致WA ...

  5. 『学了就忘』Linux基础 — 6、VMware虚拟机安装Linux系统(超详细)

    目录 1.打开VMware虚拟机软件 2.选择Linux系统的ISO安装镜像 3.开启虚拟机安装系统 (1)进入Linux系统安装界面 (2)硬件检测 (3)检测光盘 (4)欢迎界面 (5)选择语言 ...

  6. Linux文件IO操作

    来源:微信公众号「编程学习基地」 目录 文件操作 Linux文件类型 Linux文件权限 修改文件权限 Linux error 获取系统调用时的错误描述 打印错误信息 系统IO函数 open/clos ...

  7. Educational Codeforces Round 114 (Rated for Div. 2)题解

    还是常规的过了A,B,C还是在D上卡了... D. The Strongest Build 简化题意:给定你n组东西,每组东西都有\(c_i\)个装备,每个装备有一个武力值\(a_{i,j}\),要求 ...

  8. Vue脚手架最新版本安装使用

    现在很多的插件如Vant 这类的样式框架,都去兼容了Vue的3.0版本,所以我总结一下如何去简单的搭建一个Vue3.0的框架 开始 一,如何安装 在这里说明一下,Vue脚手架版本,和Vue版本是两个东 ...

  9. Excel 读写

    一.环境准备:pom.xml 导入依赖 poi-ooxml <dependencies> <dependency> <groupId>org.apache.poi& ...

  10. Screenshot 库和Collections 库

    一.screenShot 是 robot framework的标准类库,用于截取当前窗口,需要手动加载. 示例: 运行结果: 二.Collections 库 Collections 库同样为 Robo ...