CF2043C Sums on Segments

题意概要

一个数组,最多有一个数的绝对值不是 \(1\),求出所有可以得到的区间和。

思路

这里提供一个 数据结构优化查询前缀和最值 的做法。

最多有一个数的绝对值不是 \(1\),那我们可以先忽略掉这个数(记该数下标为 \(x\))。
剩下的数要么是 \(1\),要么是 \(-1\),所以我们扩展一个区间的时候,区间和的变化(或增或减)单位大小是 \(1\)
因此,我们最后的值一定是至多 \(2\) 个连续整数区间:

  • 一个是由 \(0\)(空区间) 扩展而来,并且扩展出的区间不包含 \(x\)
  • 一个是由 \(a_x\) 扩展而来。

首先考虑第一种区间,我们求出不包含 \(x\) 的所有区间和的最大值和最小值即可。
这个很明显是一个区间最值问题,但是这里涉及到区间和,所以我们先把原数组 \(a\) 进行前缀和操作转化为前缀和数组 \(pre\)
然后,我们把前缀和数组 \(pre\) 存入 线段树ST表 中维护区间前缀和最大值和最小值。
对于 \(x\) 左边的区间,我们从左到右枚举区间的左端点 \(i\),查询 \(i\)\(x\) 的区间前缀和最大值,减掉 \(pre_{i - 1}\),就可以得到以 \(i\) 作为左端点的不包含 \(x\) 的所有区间的最大区间和。最小区间和同理。
对于 \(x\) 右边的区间,我们从左到右枚举区间的左端点 \(i\),查询 \(i\)\(n\) 的区间前缀和最大值,减掉 \(pre_{i - 1}\),就可以得到以 \(i\) 作为左端点的不包含 \(x\) 的所有区间的最大区间和。最小区间和同理。
对于此类区间得到的结果取个并集,就是此类区间最后的结果。

然后是第二种区间,由 \(a_{x}\) 扩展而来。
此时这个区间一定包含 \(x\) ,因此,我们只需要以 \(x\) 为起点,分别向左向右扩展。
向左扩展变化的最小值加上向右扩展变化的最小值,就是此类区间的最小区间和。
向左扩展变化的最大值加上向右扩展变化的最大值,就是此类区间的最大区间和。

最后对两种区间求得的结果再取一个并集即可。

时间复杂度:\(O(n \log n)\)

AC CODE

// Problem: C. Sums on Segments
// Contest: Codeforces - Educational Codeforces Round 173 (Rated for Div. 2)
// URL: https://codeforces.com/contest/2043/problem/C
// Memory Limit: 256 MB
// Time Limit: 1000 ms
// 
// Powered by CP Editor (https://cpeditor.org)#include <bits/stdc++.h>
#define int long long
#define inf 2e18
#define ull unsigned long long
#define ls o << 1
#define rs o << 1 | 1using namespace std;const int N = 2e5 + 9;
int a[N];
int tmx[N << 2], tmi[N << 2];
int n;//线段树维护区间最值
void pushup(int o)
{tmx[o] = max(tmx[ls], tmx[rs]);tmi[o] = min(tmi[ls], tmi[rs]);	
}void build(int s = 1, int e = n, int o = 1)
{if(s == e)return tmx[o] = a[s], tmi[o] = a[s], void();int mid = s + e >> 1;build(s, mid, ls);build(mid + 1, e, rs);pushup(o);
}int querymx(int l, int r, int s = 1, int e = n, int o = 1)
{if(l <= s && e <= r) {return tmx[o];}int mid = s + e >> 1;int res = -inf;if(mid >= l)res = max(res, querymx(l, r, s, mid, ls));if(mid + 1 <= r)res = max(res, querymx(l, r, mid + 1, e, rs));return res;
}int querymi(int l, int r, int s = 1, int e = n, int o = 1)
{if(l <= s && e <= r) {return tmi[o];}int mid = s + e >> 1;int res = inf;if(mid >= l)res = min(res, querymi(l, r, s, mid, ls));if(mid + 1 <= r)res = min(res, querymi(l, r, mid + 1, e, rs));return res;
}void solve()
{cin >> n;for(int i = 1;i <= n;i ++)cin >> a[i];int ix = -1;for(int i = 1;i <= n;i ++)if(abs(a[i]) != 1)ix = i;for(int i = 1;i <= n;i ++)//前缀和a[i] += a[i - 1];build();//构建线段树if(ix == -1)ix = n + 1;int nomi = 0, nomx = 0;//第一类区间for(int i = 1;i <= n;i ++) {if(i == ix)continue;if(i <= ix) {nomi = min(nomi, querymi(i, ix - 1) - a[i - 1]);nomx = max(nomx, querymx(i, ix - 1) - a[i - 1]);} else {nomi = min(nomi, querymi(i, n) - a[i - 1]);nomx = max(nomx, querymx(i, n) - a[i - 1]);}}//第二类区间int hasmi = 0, hasmx = 0;if(ix <= n) {hasmi = a[ix] - a[ix - 1];hasmx = a[ix] - a[ix - 1]; int lmi = 0, lmx = 0;int rmi = 0, rmx = 0;int now = 0;for(int i = ix - 1;i;i --) {now += a[i] - a[i - 1];lmi = min(lmi, now);lmx = max(lmx, now);}now = 0;for(int i = ix + 1;i <= n;i ++) {now += a[i] - a[i - 1];rmi = min(rmi, now);rmx = max(rmx, now);}hasmi = hasmi + lmi + rmi;hasmx = hasmx + lmx + rmx;} set<int> ans;for(int i = nomi;i <= nomx;i ++)ans.insert(i);for(int i = hasmi;i <= hasmx;i ++)ans.insert(i);cout << ans.size() << '\n';for(auto &i : ans)cout << i << ' ';cout << '\n';
}signed main()
{ios::sync_with_stdio(0),cin.tie(0),cout.tie(0);int t = 1;cin >> t;while(t --)solve();return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/859536.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

PostgreSQL 初始化配置设置

title: PostgreSQL 初始化配置设置 date: 2024/12/27 updated: 2024/12/27 author: cmdragon excerpt: PostgreSQL是一款广泛应用于企业级应用、数据仓库以及Web应用程序的强大数据库管理系统。在完成数据库的安装后,进行合理而有效的初始配置是确保数据库性能和安全性的关键…

Chrome浏览器不太愿意使用http3/quic怎么办?如何强制让Chrome开启http3/quic

1. 确认你要访问的网站支持h3 一定要有Alt-svc标头2. 配置Chrome启动参数,对指定网站强制开启Http3 右键点击桌面Chrome图标,选择属性,在Chrome.exe后添加启动参数,网站地址替换为你想要开启的网站地址。 chrome.exe --origin-to-force-quic-on=localhost:6001 https://loc…

postgis和postgresql学习

一、两者关系 postgresql本质上还是sql驱动的数据库,和mysql、sqlserver等是性质相同的 postgis本质上是postgresql的插件,可以将postgresql这种关系型数据库改造成空间数据库 二、环境配置 postgresql:https://www.enterprisedb.com/downloads/postgres-postgresql-downloa…

并发编程 - 死锁的产生、排查与解决方案

多线程中死锁因资源争夺形成循环等待,必要条件为互斥、占有并等待、不可剥夺、循环等待。可用VisualStudio等工具排查,解决办法包括顺序加锁、尝试锁、超时机制、避免嵌套使用锁。在多线程编程中,死锁是一种非常常见的问题,稍不留神可能就会产生死锁,今天就和大家分享死锁…

一维数组、多维数组、Array(deepToString sort fill binarySearch)方法2024122620241226

数组20241226 [数组详情](深入理解 Java 数组 - 静默虚空 - 博客园)什么是数组: 数组是相同类型数据的有序集合注意:必须是相同数据数组描述的是相同类型的若干个数据,按照一定的先后次序排列组合而成其中,每一个数据称作一个数组元素。 每个数组元素可以通过一个下标来访问它…

《计算机组成及汇编语言原理》阅读笔记:p116-p120

《计算机组成及汇编语言原理》学习第 7 天,p116-p120 总结,总计 5 页。 一、技术总结 1.CPU优化 (1)increase overall performance number 例如:16位电脑提升到32位电脑。 (2)multiprocessing One way to make computers more useful is to allow them to run more than on…

波折重重:Linux实时系统Xenomai宕机问题的深度定位过程

本文将带您深入了解一个与之相关的真实事故现场及其问题定位过程,波折重重,其中的xenomai问题定位思路具有一定借鉴意义,希望对你定位xenomai问题有所帮助。目录一 前言二 背景三 原因分析及措施硬件原因应用软件操作系统四 分析定位转机拨云见雾irq计数Schedstatcoreclk现象…

Java面向对象程序设计复习总结

作者:高世栋 学号:202302151071 一、第一章:初识Java与面向对象程序设计Java简介:Java是一种面向对象的程序设计语言,具有跨平台、安全性高、可移植性强等特点。面向对象程序设计概述:面向对象是一种程序设计思想,将现实世界的事物抽象为对象,通过对象之间的交互来完…

[Paper Reading] StegoType: Surface Typing from Egocentric Cameras

目录StegoType: Surface Typing from Egocentric CamerasTL;DRData数据采集设备开环数据收集闭环数据收集数据容错机制OracleMethodInput FeaturesBackboneDataLossExperiment效果可视化总结与思考相关链接Related works中值得深挖的工作资料查询 StegoType: Surface Typing fr…

JVM实战—2.JVM内存设置与对象分配流转

大纲 1.JVM内存划分的原理细节 2.对象在JVM内存中如何分配如何流转 3.部署线上系统时如何设置JVM内存大小 4.如何设置JVM堆内存大小 5.如何设置JVM栈内存与永久代大小 6.问题汇总1.JVM内存划分的原理细节 (1)背景引入 (2)大部分对象的存活周期都是极短的 (3)少数对象是长期存活…

【Obsidian】 博客园插件

搬运 原文作者:ZhangBlog 出处:https://www.cnblogs.com/aaalei/p/17926199.html由于 Markdown 语法的便捷性, 我们从繁重的排版布局工作中解脱出来, 越来越多的人开始接受这种写作方式, 该插件可以将你的 md 笔记, 方便的同步到博客园中, 即使你是使用的本地图片, 也无须担心…

【数据分析】如何构建数据分析体系?

一、数据分析体系的重要性二、如何搭建数据分析体系三、数据分析体系如何量体裁衣编者荐语: 很详细 以下文章来源于ruby的数据漫谈 ,作者ruby 摘要:在当今数字化时代,数据已成为企业决策和发展的重要依据。构建一个完善的数据分析体系,能够帮助企业从海量数据中挖掘价值,…