P8684 [蓝桥杯 2019 省 B] 灵能传输 题解

Part 1 提示

  • 题目传送门
  • 欢迎大家指出错误并私信这个蒟蒻
  • 欢迎大家在下方评论区写出自己的疑问(记得 @ 这个蒟蒻)

Part 2 更新日志

  • 2023-06-20 21:46 文章完成
  • 2023-07-03 08:57 文章通过审核
  • 2023-08-21 18:14 更改了文章格式,使文章看起来更加美观

Part 3 背景

这是这个蒟蒻做了将近 \(4\) 天的题目,所以来写篇题解纪念一下。

Part 4 解析

本题涉及到了 \(3\) 种算法:前缀和,排序以及贪心

(1)前缀和

本题实际上要求通过某种灵能传输可以使得该序列的最大值最小。而由前缀和可知,当某一个前缀和序列保持有序(或前缀和序列表示的函数单调)时,其 \(\max(s[i]-s[i-1])\) 的最大值可以达到最小。

通过对几个样例的观察我们不难发现:

1.当 \(a[i]>0\) 时,若 \(a[i-1]=a[i-1]+a[i]\),则 \(s[i-1]\) 等于原来的 \(s[i]\)。

2.若 \(a[i]=a[i]-2a[i]\),则原 \(s[i-1]=s[i-1]+s[i]\)。

3.现 \(s[i]=\) 现 \(s[i-1]-a[i]=\) 原 \(s[i]-a[i]=\) 原 \(s[i-1]\)。

这意味着除了 \(s[0]\) 和 \(s[n]\) 以外,\(1\sim n\) 的任何 \(s[i]\) 都可以进行互相交换,从而得到一个有序序列。而 \(a[i]=s[i]-s[i-1]\) 也就意味着可以通过交换 \(s[i]\) 的方式得到灵能传输后的最终结果。

(2)排序

for (int i = 1; i <= n; i++) {
scanf("%d", &a[i], s[i] = s[i - 1] + a[i]);
}
sort(s + 1, s + 1 + n);

当然,如果 \(s[0]\) 和 \(s[n]\) 也可以正常交换,则只需要将整个前缀和序列进行排序,即可直接得到一个单调函数,那么本题的推导到这一步就可以结束了,可以通过直接计算 \(\max(s[i]-s[i-1])\) 的值获得最大值和最小值。但问题就在于 \(s[0]\) 和 \(s[n]\),即最终得到的序列不一定是单调的,所以接下来就要通过一系列操作解决序列不单调的问题。

(3)贪心

通过上述的分析可以得知,想要求出本题的最优解就是使得所求序列尽可能保持单调。通过画图可知,在两个端点无法移动的条件下,在对于整个前缀和序列进行排序时,总能得到一个拥有两个拐点且中间部分保持单调的函数。此时就应该往贪心上思考,即当一条有两个拐点的曲线的重叠部分最小时单调部分最多,而一条曲线符合下列情况时符合要求。

①左端点小于右端点,即 \(s[0]<s[n]\)。在记录 \(s_0\) 和 \(s_n\) 的值时需要进行一次判定,如果得到的左端点比右端点大,那么就将这两个端点交换(尽量保证得到的函数是一个中部递增的单调函数,其目的是将得到的所有函数都变成中部递增函数,这样就可以少算至少一半的数据)。

if (s0 > sn) {
swap(s0, sn);
}

②极小值在极大值左边(刚刚的情况中,要求得到的函数一定是中部递增的,因此不仅需要控制函数中部的递增,还要控制最大值和最小值以使得中部函数递增)。这就要求在后续选点时应遵循 \(s[0]\) 向左取,\(s[n]\) 向右取,因为这样才能取得两边的极值。

因为已经将两个端点确定并保证了两者的顺序,也对前缀和序列进行了升序处理,于是此时得到了一个存放着递增的前缀和序列的有序数组(左右端点的位置已经发生改变,情况①中已经记录了两者位置)。

接下来需要从左端点的位置向左依次取点,从右端点的位置向右依次取点(从左端点向左依次取点并取得前缀和序列的最小值,从右端点向右依次取点并取得前缀和序列的最大值)。此时通过画图可以求得函数为两个端点有拐点且中部有序递增的函数。

int l = 0, r = n - 1;
for (int i = s0; i >= n; i -= 2) {
f[l++] = s[i];
st[i] = true;
}
for (int i = sn; i <= n; i += 2) {
f[r--] = s[i];
st[i] = true;
}
for (int i = 0; i <= n; i++) {
if (st[i] == false) {
f[l++] = s[i];
}
}

因为图像中有两个拐点而且会形成两个重叠部分,所以想要得到最优解,就要使得求得的函数图像中的递增部分尽可能地多,这样拐点处的图像就会尽可能地少,即可保证序列 \(f\) 为重叠部分最小的前缀和序列。

在通过特定规则将所有点都遍历完毕后,此时已经得到最优解的图像(前缀和序列)。最后就是求出所有前缀和表示的灵能值中的最大者(一定为正),该灵能值便是最终答案。

int res = 0;
for (int i = 1; i <= n; i++) {
res = max(res, abs(f[i] - f[i - 1]));
}

\(res\) 即为所求结果。

Part 5 代码

#include <bits/stdc++.h>
using namespace std;
const int N = 3e5 + 10;
typedef long long ll;
ll a[N];
ll s[N];
ll f[N];
bool st[N];
int main() {
int t;
scanf("%d", &t);
while (t--) {
int n;
scanf("%d", &n);
s[0] = 0;
for (int i = 1; i <= n; i++) {
scanf("%lld", &a[i]);
s[i] = s[i - 1] + a[i];
}
ll s0 = s[0];
ll sn = s[n];
if (s0 > sn) {
swap(s0, sn);
}
sort(s, s + 1 + n);
for (int i = 0; i <= n; i++) {
if (s0 == s[i]) {
s0 = i;
break;
}
}
for (int i = 0; i <= n; i++) {
if (sn == s[i]) {
sn = i;
break;
}
}
memset(st, false, sizeof st);
int l = 0, r = n;
for (int i = s0; i >= 0; i -= 2) {
f[l++] = s[i];
st[i] = true;
}
for (int i = sn; i <= n; i += 2) {
f[r--] = s[i];
st[i] = true;
}
for (int i = 0; i <= n; i++) {
if (st[i] == false) {
f[l++] = s[i];
}
}
ll res = 0;
for (int i = 1; i <= n; i++) {
res = max(res, abs(f[i] - f[i - 1]));
}
printf("%lld\n", res);
}
return 0;
}

P8684 [蓝桥杯 2019 省 B] 灵能传输 题解的更多相关文章

  1. 第十届蓝桥杯2019年C/C++ 大学B组省赛试题

    2019年第十届蓝桥杯大赛软件类省赛C/C++大学B组 试题 A:组队 本题总分:5分 [问题描述] 作为篮球队教练,你需要从以下名单中选出 1号位至 5号位各一名球员, 组成球队的首发阵容. 每位球 ...

  2. 第十届蓝桥杯2019年C/C++ 大学A组省赛试题

    2019年蓝桥杯第十届软件类省赛 C/C++ 大 学 A 组 试题 A: 平方和 本题总分:5 分 [问题描述] 小明对数位中含有 2.0.1.9 的数字很感兴趣,在 1 到 40 中这样的数包括 1 ...

  3. 【备考06组01号】第四届蓝桥杯JAVA组A组国赛题解

    1.填算式 (1)题目描述     请看下面的算式:     (ABCD - EFGH) * XY = 900     每个字母代表一个0~9的数字,不同字母代表不同数字,首位不能为0.     比如 ...

  4. 2019年第十届蓝桥杯c++A组java/c++组题解

    #include<iostream> #include<vector> using namespace std; vector <int > vec; long l ...

  5. 今日学习——蓝桥杯 2019年 C语言 B组

    1.手淦(亲身体验,,,没啥大用,最终还是代码) 2.代码(下面是我看其他博主代码答案能看的懂的....具体的可以直接去下面的网址看) https://blog.csdn.net/qq_4452491 ...

  6. 2019 蓝桥杯国赛 B 组模拟赛 题解

    标签 ok #include<bits/stdc++.h> using namespace std; /* 求阶乘 去除尾部0 每次求阶乘时:结果去除尾0,并对 1e6取余 */ type ...

  7. 蓝桥杯2019初赛]迷宫(dfs版本)

    传送门 大意: 题目的意思还是模板的搜索,不同的是我们要记录路径了,而且是最短字典序最小的路径. 思路: 1.对于字典序最小,也就是说我们要尽量先往下走,然后是左- 这个很简单,因为在dfs中是顺序枚 ...

  8. P8701 [蓝桥杯 2019 国 B] 第八大奇迹

    简要题意 你需要维护一个长度为 \(L\) 的序列 \(a\),初始时全部都是 \(0\),有 \(N\) 个操作,支持: C p x,将 \(a_p\) 修改为 \(x\). Q a b,输出 \( ...

  9. 2015年蓝桥杯B组C/C++决赛题目

    2015年第六届蓝桥杯B组C/C++国赛题目 点击查看2015年第六届蓝桥杯B组C/C++国赛题解     1.积分之迷 小明开了个网上商店,卖风铃.共有3个品牌:A,B,C. 为了促销,每件商品都会 ...

  10. 2018年蓝桥杯B组C/C++决赛题目

    自己的博客排版,自我感觉略好一点. 先放上题目. 点击查看2018年蓝桥杯B组C/C++决赛题目题解     1.换零钞 x星球的钞票的面额只有:100元,5元,2元,1元,共4种. 小明去x星旅游, ...

随机推荐

  1. 《最新出炉》系列初窥篇-Python+Playwright自动化测试-4-playwright等待浅析

    1.简介 在介绍selenium的时候,宏哥也介绍过等待,是因为在某些元素出现后,才可以进行操作.有时候我们自己忘记添加等待时间后,查了半天代码确定就是没有问题,奇怪的就是获取不到元素.然后搞了好久, ...

  2. ARM Trusted Firmware——编译选项(二)

    @ 目录 1. 常用部分 2. 安全相关 2.1 签名 2.2 加密 2.3 哈希 2.4 中断 3.GICv3驱动程序选项 4. 调试选项 1. 常用部分 编译选项 解释 BL2 指定生成fip文件 ...

  3. MAUI Blazor 显示本地图片的新思路

    前言 好久没写文章了,水一篇 关于MAUI Blazor 显示本地图片这个问题,有大佬发过了. 就是 token 大佬的那篇 Blazor Hybrid (Blazor混合开发)更好的读取本地图片 主 ...

  4. 编码技巧 --- 使用dynamic简化反射

    引言 dynamic 是 Framework 4.0 就出现特性,它的出现让 C# 具有了弱语言类型的特性.编译器在编译的时候不再对类型进行检查,默认 dynamic 对象支持开发者想要的任何特性. ...

  5. 记一次 .NET 某物流API系统 CPU爆高分析

    一:背景 1. 讲故事 前段时间有位朋友找到我,说他程序CPU直接被打满了,让我帮忙看下怎么回事,截图如下: 看了下是两个相同的程序,既然被打满了那就抓一个 dump 看看到底咋回事. 二:为什么会打 ...

  6. python入门,一篇就够了

    python规范 函数必须写注释:文档注释格式'''注释内容''' 参数中的等号两边不要用空格 相邻函数用两个空行隔开 小写 + 下划线 函数名 模块名 实例名 驼峰法 类名 tips # 一行代码太 ...

  7. 小白也能搞定!Windows10上CUDA9.0+CUDNN7.0.5的完美安装教程

    前言: 为什么要在本地电脑安装 CUDA,CUDA 是什么的,用来做什么?我想,点击标题进来的小伙伴,应该都清楚这些.不管你是用来做什么,或者跟我一样为了跑 Tensorflow 的 Object D ...

  8. [kvm]cpu内存硬盘配置

    修改CPU配置 如果配置了最大CPU # 临时 virsh setvcpus test 2 # 永久 virsh setvcpus test 2 --config 热增加虚拟机的CPU数后,使用lsc ...

  9. c#如何使用WASM跨语言调用?

    介绍Wasm(WebAssembly) WebAssembly(简称Wasm)是一种用于基于堆栈的虚拟机的二进制指令格式.Wasm被设计为编程语言的可移植编译目标,支持在web上部署客户端和服务器应用 ...

  10. php批量同步数据

    php批量同步流程 首先分页获取数据 创建临时表 批量添加数据 备份原表 删除原表 修改临时表表名改为原表 代码 1 <?php 2 3 class Stock{ 4 5 private $da ...