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

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

时间:2024-05-17 22:56:39浏览次数:32  
标签:灵能 前缀 int 题解 蓝桥 端点 序列 gets 函数

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

(1)前缀和
本题实际上要求通过某种灵能传输可以使得该序列的最大值最小。而由前缀和可知,当某一个前缀和序列保持有序(或前缀和序列表示的函数单调)时,其 \(s[i] - s[i - 1]\) 的最大值可以达到最小。
通过对几个样例的观察我们不难发现:
1.当 \(a[i] > 0\) 时,若 \(a[i - 1] \gets a[i - 1] + a[i]\),则 \(s[i - 1]\) 等于原来的 \(s[i]\)。
2.若 \(a[i] \gets a[i] - 2 \times a[i]\),则原 \(s[i - 1] \gets s[i - 1] + a[i]\)。
3.现 $s[i] \gets $ 现 $s[i - 1] - a[i] \gets $ 原 \(s[i] - a[i] \gets\) 原 \(s[i - 1]\)。
这意味着除了 \(s[0]\) 和 \(s[n]\) 以外,\(1 \sim n\) 的任何 \(s[i]\) 都可以进行互相交换,从而得到一个有序序列。而 \(a[i] \gets 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\) 即为所求结果。

代码如下:

#include <bits/stdc++.h>

using namespace std;

const int N = 3e5 + 10;

typedef long long ll;

int t;

ll a[N], s[N], f[N];

bool st[N];

void work()
{
    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;
}

int main()
{
    scanf("%d", &t);
    while (t--)
    {
        work();
    }
    return 0;
}

标签:灵能,前缀,int,题解,蓝桥,端点,序列,gets,函数
From: https://www.cnblogs.com/cq-irritater/p/18198853/solution-luogu-p8684

相关文章

  • P9686 Judg. 题解
    题目传送门一道简单模拟。这道题最简单的方法就是直接在for循环中判断题目给出状态是否为AC,如果不是则输出当前\(i\)的值,否则就不输出。#include<bits/stdc++.h>usingnamespacestd;constintMAXN=1e5+10;intn;stringa[MAXN];intmain(){ scanf("%......
  • P8741 [蓝桥杯 2021 省 B] 填空问题 题解
    题目传送门试题A:空间【解析】本题考察计算机存储的基础知识,只要掌握空间存储的换算方法,就能够算出答案。【程序】#include<bits/stdc++.h>usingnamespacestd;intmain(){printf("%d\n",256*8/32*1024*1024);return0;}【答案】67108864......
  • P8679 [蓝桥杯 2019 省 B] 填空问题 题解
    题目传送门试题A:组队【解析】本题是一道经典的DFS搜索题,每次对各号位的选手进行DFS,找到各号位上有成绩的选手时再进行进一步搜索即可。【程序】#include<bits/stdc++.h>usingnamespacestd;intteam[20][6];intmax_sum;boolvis[20];voiddfs(intu,intsu......
  • P9517 drink 题解
    题目传送门这道题考场上用的查找做的。先用一个结构体分别表示firstlast,然后进行查找即可,两个for循环分别计算出first和last,最后计算它们的差值。(注意,计算差值时要加1)然后你就会发现一个问题:只有\(90\)分。所以我们来思考一下哪里出现了问题。你会发现:如果全都是......
  • Codeforces 959B Mahmoud and Ehab and the message 题解
    题目简述小A想要给他的朋友小B发送了一条有$m$个单词的消息。他们的语言由编号从$a_1$到$a_n$的$n$个单词组成。一些单词具有相同的意思,因此存在$k$个单词组,其中每个组中的所有单词具有相同的意思。小A知道第$i$个单词可以以成本$m_i$发送。对于他的每个消息......
  • Codeforces 1113B Sasha and Magnetic Machines 题解
    题目简述有一个长度为$n$的正整数序列。你可以对这个数列进行最多$1$次的如下操作:选择两个数$i$和$j$,其中$1\leqi,j\leqn$并且$i\neqj$,并选择一个可以整除$a_i$的正整数$x$,然后将$a_i$变为$\frac{a_i}{x}$,将$a_j$变为$a_j\cdotx$。问你操作后,该序......
  • Codeforces 1037C Equalize 题解
    题目描述给定两个长度为$n$的$01$序列$a,b$。每次可以执行如下操作:在$a$中选择一个位置$p$,将$a_p$变为$1-a_p$,代价是$1$。在$a$中选择两个位置$p,q$,将$a_p$和$a_q$互换,代价是$\lvertq-p\rvert$。问最少需要多少代价才能将$a$变成$b$。题目分析......
  • CF1884D Counting Rhyme 题解
    题目链接:CF或者洛谷给个莫反题解,讲讲常规套路题目要求满足没有\(a_k\mida_i与a_k\mida_j\)的\((i,j)\)的对数,显然即不存在\(a_k\mid\gcd(a_i,a_j)\)。稍微拓展下,如果不存在整除多个数,那么显然不整除它们的\(\gcd\)即可,因为它们的公因数即为满足的最大数,如果为......
  • ArchLinux/Manjaro升级到6.9内核后的问题解决
    1.KDEWallet系统配置---个性化---账户详细信息---kde钱包1.需要启用kde钱包子系统,否则无法正常使用记录的账号密码信息2.右下角,调用钱包管理器,修改密码,设置为空密码至此,开机需要输密码连接kdewallet的应用不需要输入即可密码连接2.更新archcn-keyring报GPG错误解决:sudopa......
  • P10466的题解
    (一)出门左转P3369。只需要同时记录原本属于哪一位即可。这里给出Splay做法。(二)AC代码。建议自己打一遍巩固印象。虽然我是直接拉过来的。#include<bits/stdc++.h>#defineintlonglongusingnamespacestd;inlineintread(){intx=0,f=1;charc=getchar();wh......