CF1333C Eugene and an array
提供一种复杂度为
考虑数据结构加二分。
如果我们指定某个区间的
所以可以考虑枚举
可以将原序列做一次前缀和,那么如果一个区间内有和为 vector
上二分可以在线处理。
#pragma GCC optimize("-Ofast")
#include <iostream>
#include <cstdio>
#include <algorithm>
#include <map>
#include <vector>
#include <cstring>
using namespace std;
#define int long long
const int N = 2e5 + 5, M = 25;
int a[N], n, pre[N], p[N], LG2[N], s2[N];
vector<int> b;
map<int, vector<int> > pee;
void Init()
{
LG2[1] = 0;
LG2[2] = 1;
for (int i = 3; i < N; i++)
{
LG2[i] = LG2[i / 2] + 1;
}
}
int f[N][M];
inline int query(int l, int r)
{
int x = LG2[r - l + 1];
return max(f[l][x], f[r - (1LL << x) + 1][x]);
}
signed main()
{
Init();
scanf("%lld", &n);
int ret = n * (n - 1) / 2 + n;
for (int i = 1; i <= n; i++)
{
scanf("%lld", &a[i]);
if (a[i] == 0) ret--;
a[i] += a[i - 1];
s2[i] = a[i];
b.push_back(a[i]);
pee[a[i]].push_back(i);
}
sort(b.begin(), b.end());
b.erase(unique(b.begin(), b.end()), b.end());
for (int i = 1; i <= n; i++) a[i] = lower_bound(b.begin(), b.end(), a[i]) - b.begin() + 1;
for (int i = 1; i <= n; i++)
{
pre[i] = p[a[i]];
p[a[i]] = i;
}
for (int i = 1; i <= n; i++)
{
f[i][0] = pre[i];
}
for (int j = 1; j < M; j++)
{
for (int i = 1; i + (1ll << j) - 1 <= n; i++)
{
f[i][j] = max(f[i][j - 1], f[i + (1ll << (j - 1))][j - 1]);
}
}
for (int i = 1; i < n; i++)
{
int l = i, r = n + 1, place = -1;
while (l + 1 < r)
{
int mid = l + r >> 1;
int x = query(i, mid);
if ((x >= i - 1 && x != 0) || upper_bound(pee[-s2[i - 1]].begin(), pee[-s2[i - 1]].end(), mid) - lower_bound(pee[-s2[i - 1]].begin(), pee[-s2[i - 1]].end(), i) >= 1) r = mid, place = mid;
else l = mid;
}
//printf("%lld %lld %lld\n", i, place, query(i, place));
if (place != -1) ret -= (n - place + 1);
}
printf("%lld\n", ret);
return 0;
}
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 分享一个免费、快速、无限量使用的满血 DeepSeek R1 模型,支持深度思考和联网搜索!
· 基于 Docker 搭建 FRP 内网穿透开源项目(很简单哒)
· ollama系列01:轻松3步本地部署deepseek,普通电脑可用
· 25岁的心里话
· 按钮权限的设计及实现