[题解] P8684 [蓝桥杯 2019 省 B] 灵能传输
本题涉及到了
(1)前缀和
本题实际上要求通过某种灵能传输可以使得该序列的最大值最小。而由前缀和可知,当某一个前缀和序列保持有序(或前缀和序列表示的函数单调)时,其
通过对几个样例的观察我们不难发现:
1.当
2.若
3.现
这意味着除了
(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);
当然,如果
(3)贪心
通过上述的分析可以得知,想要求出本题的最优解就是使得所求序列尽可能保持单调。通过画图可知,在两个端点无法移动的条件下,在对于整个前缀和序列进行排序时,总能得到一个拥有两个拐点且中间部分保持单调的函数。此时就应该往贪心上思考,即当一条有两个拐点的曲线的重叠部分最小时单调部分最多,而一条曲线符合下列情况时符合要求。
①左端点小于右端点,即
if (s0 > sn) { swap(s0, sn); }
②极小值在极大值左边(刚刚的情况中,要求得到的函数一定是中部递增的,因此不仅需要控制函数中部的递增,还要控制最大值和最小值以使得中部函数递增)。这就要求在后续选点时应遵循
因为已经将两个端点确定并保证了两者的顺序,也对前缀和序列进行了升序处理,于是此时得到了一个存放着递增的前缀和序列的有序数组(左右端点的位置已经发生改变,情况①中已经记录了两者位置)。
接下来需要从左端点的位置向左依次取点,从右端点的位置向右依次取点(从左端点向左依次取点并取得前缀和序列的最小值,从右端点向右依次取点并取得前缀和序列的最大值)。此时通过画图可以求得函数为两个端点有拐点且中部有序递增的函数。
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]; } }
因为图像中有两个拐点而且会形成两个重叠部分,所以想要得到最优解,就要使得求得的函数图像中的递增部分尽可能地多,这样拐点处的图像就会尽可能地少,即可保证序列
在通过特定规则将所有点都遍历完毕后,此时已经得到最优解的图像(前缀和序列)。最后就是求出所有前缀和表示的灵能值中的最大者(一定为正),该灵能值便是最终答案。
int res = 0; for (int i = 1; i <= n; i++) { res = max(res, abs(f[i] - f[i - 1])); }
代码如下:
#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; }
本文作者:cq_irritater
本文链接:https://www.cnblogs.com/cq-irritater/p/18198853/solution-luogu-p8684
版权声明:本作品采用知识共享署名-非商业性使用-禁止演绎 2.5 中国大陆许可协议进行许可。
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步