首页 > 其他分享 >P8684 [蓝桥杯 2019 省 B] 灵能传输 题解

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

时间:2023-10-15 10:33:18浏览次数:37  
标签:灵能 前缀 int 题解 蓝桥 sn 端点 序列 函数

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;
}

标签:灵能,前缀,int,题解,蓝桥,sn,端点,序列,函数
From: https://www.cnblogs.com/szyawa/p/solution-p8684.html

相关文章

  • 算法题解——买卖股票的最佳时机
    解题思路先考虑最简单的「暴力遍历」,即枚举出所有情况,并从中选择最大利润。设数组prices的长度为n,由于只能先买入后卖出,因此第1天买可在未来n−1天卖出,第2天买可在未来n-2天卖出……以此类推,共有\[(n-1)+(n-2)+\cdots+0=\frac{n(n-1)}{2}\]种情况,时间复......
  • Atcoder Beginner Contest 324 F Beautiful Path 题解-分数规划
    为了更好的阅读体验,请点击这里分数规划小技巧:尽可能将式子写成存在某种取值,使得不等式成立的形式。不然可能需要绕几个弯才能想出来。题目链接题目大意:给出一个DAG,每条边有一个\(b_i,c_i\),保证从编号小的边向编号大的边连边,且\(1\)到\(n\)必有路径,求\(1\)到\(n\)......
  • CF1204D2 Kirk and a Binary String (hard version) 题解
    CF1204D2KirkandaBinaryString(hardversion)题解分析先来分析\(01\)串的最长不下降子序列。全是\(0\)显然是不下降的,如果中间出现一个\(1\),为了维护不下降的性质,后面就只能全是\(1\)。一句话概括一下,\(0\)后面能跟\(0,1\),\(1\)后面只能跟\(1\)。现在来分析这......
  • 题解 [ABC258G] Triangle
    题目链接\(\rmO(n^3)\)枚举\(i,j,k\)的算法是显然的。考虑优化掉一个\(n\),如果枚举\(i,j\),那么显然需要找出有多少个\(k\)同时满足\(a_{i,k}=a_{j,k}=1\),我们可以将\(a_i\)和\(a_j\)看作两个二进制数,那么同时等于\(1\)的位置就是并起来等于\(1\)的位置,\(bitset......
  • 观光奶牛 详细题解
    #T3#SPFA判断正/负环#二分查找为啥现在突然发出来:翻自个笔记发现这篇写的挺好hhh361.观光奶牛-AcWing题库给定一张\(L\)个点、\(P\)条边的有向图,每个点都有一个权值\(f[i]\),每条边都有一个权值\(t[i]\)。求图中的一个环,使“环上各点的权值之和”除以“环上各边的......
  • [题解]AT_abc153_f [ABC153F] Silver Fox vs Monster
    模拟赛最后\(15\)分钟想到的做法。思路首先有一个显然的贪心策略:我们放炸弹的地方要尽可能的使这个炸弹能影响到更多的怪上。那么我们可以将对于一个怪\(i\)能够影响到它的区间表示出来\([\max(1,l_i-d),a_i+r]\)。然后将这些区间排个序,可以粗略画出这样的图:根据上......
  • P1084疫情控制 题解
    P1084疫情控制前言:这题思路不难,实现稍微有点难。总体来说,不算特别难的那种紫题,建议评蓝。题目描述给定一些点,用这些点来切断根节点到所有叶子节点的路径,可以移动这些点,不同的点可以同时移动,求时间最少。思考过程不同的点可以同时移动:看到这里,我们可以转化一下题目:给定......
  • [AGC033C] Removing Coins题解
    思路可以看出,每次对一个点\(u\)操作一次,就相当于删除以\(u\)为根的所有叶节点。当然我们还是没有什么思路,我们可以想简单一点:在一条链上的情况。如果\(u\)是链的端点:以\(u\)为根节点的叶节点只有一个,所以链的长度减一。如果\(u\)不是链的端点:以\(u\)为根节点......
  • P1612 [yLOI2018] 树上的链 题解
    思路看到条件\(2\),我们得知:这个节点对应的最长链,一定在这个节点到根节点的简单路径上。所以我们记录\(1\)到\(i\)之间的权值和,记为\(sum_i\)。因为权值是正整数,所以满足单调性,可以二分。如何二分路径上的点呢?我们维护一个与当前dfs同步的栈,记录从根节点到当前节点的简......
  • [ARC116C] Multiple Sequences题解
    思路我们可以很好的想到一种\(O(nm)\)的dp:状态:\(dp_{i,j}\)为搜到第\(i\)个,最后一个数是\(j\)的方案数。转移:\(dp_{i,j}=\displaystyle\sum_{k|j,k\not=j}dp_{i-1,k}\)当然这是会超时的。我们换一种思路,我们先枚举最后一个数,再计算方案数。这有个好处,我们缩小......