【题目大意】

小奇和魔法猪要竞选膜钟国的总统。

有 $n$ 个选民,编号为$1...n$,他们中有的人支持小奇,有的人支持魔法猪,还有的人保持中立。

现在你需要把选民分成若干个区间,每个区间的长度在$[l,r]$中。如果一个区间中支持小奇的人比支持魔法猪的人多,那么小奇得一票,一个区间中支持魔法猪的人比支持小奇的人多,那么魔法猪得一票。

小奇想要赢得选举,于是它请你合理地分区间,使得它获得的票数减去魔法猪获得的票数最大。如果你帮它解决了这个问题,它会送你这道题的题解。

$n \leq 10^6$

【题解】

考虑暴力dp,复杂度为$O(n^2)$。

然后考虑优化暴力,每次相当于区间往右移动一格,相当于加入1个数,减去1个数,这个可以用线段树来维护

如果普通写法需要开单调队列,我们可以把下标映射到位置,这样就能执行减去1个数的操作了。

复杂度$O(nlogn)$

# include <vector>
# include <stdio.h>
# include <string.h>
# include <iostream>
# include <algorithm> using namespace std; typedef long long ll;
typedef unsigned long long ull;
typedef long double ld; const int M = 5e5 + , N = 1e6 + , inf = 1e9, SN = * + ; inline int getint() {
int x = , f = ; char ch = getchar();
while(!isdigit(ch)) {
if(ch == '-') f = ;
ch = getchar();
}
while(isdigit(ch)) x = (x<<) + (x<<) + ch - '', ch = getchar();
return f ? x : -x;
} int n, L, R, a[N], s[N], f[N], pos[N], idx;
vector<int> v[N + N]; int l[N], r[N]; struct SMT {
int w[N << ];
# define ls (x<<)
# define rs (x<<|)
inline void up(int x) {
w[x] = max(w[ls], w[rs]);
}
inline void build(int x, int l, int r) {
w[x] = -inf;
if(l == r) return ;
int mid = l+r>>;
build(ls, l, mid); build(rs, mid+, r);
}
inline void edt(int x, int l, int r, int pos, int del) {
if(l == r) {
w[x] = del;
return ;
}
int mid = l+r>>;
if(pos <= mid) edt(ls, l, mid, pos, del);
else edt(rs, mid+, r, pos, del);
up(x);
}
inline int query(int x, int l, int r, int L, int R) {
if(L > R) return -inf;
if(L <= l && r <= R) return w[x];
int mid = l+r>>;
if(R <= mid) return query(ls, l, mid, L, R);
else if(L > mid) return query(rs, mid+, r, L, R);
else return max(query(ls, l, mid, L, mid), query(rs, mid+, r, mid+, R));
}
# undef ls
# undef rs
}T; inline void ins(int i) {
// cout << "add " << pos[i] << endl;
T.edt(, , n, pos[i], f[i]);
} inline void del(int i) {
T.edt(, , n, pos[i], -inf);
} int main() {
freopen("election.in", "r", stdin);
freopen("election.out", "w", stdout);
n = getint(), L = getint(), R = getint();
for (int i=; i<=n; ++i) a[i] = getint();
for (int i=; i<=n; ++i) s[i] = s[i-] + a[i];
// for (int i=0; i<=n; ++i) cout << s[i] << ' '; cout << endl; for (int i=; i<=n; ++i) v[s[i] + n].push_back(i);
for (int i=; i<=n+n; ++i) {
int lst = idx;
for (int j=; j<v[i].size(); ++j)
pos[v[i][j]] = idx++;
for (int j=; j<v[i].size(); ++j)
l[v[i][j]] = lst, r[v[i][j]] = idx - ;
}
// for (int i=0; i<=n; ++i) cout << pos[i] << ' ' << l[i] << ' ' << r[i] << endl; cout << endl; f[] = ;
T.build(, , n);
for (int i=; i<=n; ++i) {
if(i - R - >= ) del(i - R - );
if(i - L >= ) ins(i - L);
int ta = T.query(, , n, , l[i]-), tb = T.query(, , n, r[i]+, n), tc = T.query(, , n, l[i], r[i]);
if(ta != -inf) ta ++;
if(tb != -inf) tb --;
// cout << i << "???" << l[i]-1 <<endl;
// cout << i << ' ' << ta << ' ' << tb << ' ' << tc << endl;
f[i] = max(ta, max(tb, tc));
} // }
if(f[n] == -inf) cout << "Impossible";
else cout << f[n];
return ;
}

省队集训 Day5 选举的更多相关文章

  1. FJ省队集训DAY5 T1

    思路:考试的时候打了LCT,自以为能过,没想到只能过80.. 考完一想:lct的做法点数是100W,就算是nlogn也会T. 讲一下lct的做法把:首先如果一条边连接的两个点都在同一个联通块内,那么这 ...

  2. JS省队集训记

    不知不觉省队集训已经结束,离noi也越来越近了呢 论考前实战训练的重要性,让我随便总结一下这几天的考试 Day 1 T1 唉,感觉跟xj测试很像啊?meet in middle,不过这种题不多测是什么 ...

  3. HN2018省队集训

    HN2018省队集训 Day1 今天的题目来自于雅礼的高二学长\(dy0607\). 压缩包下载 密码: 27n7 流水账 震惊!穿着该校校服竟然在四大名校畅通无阻?霸主地位已定? \(7:10\)从 ...

  4. 2019暑期金华集训 Day5 树上数据结构

    自闭集训 Day5 树上数据结构 前置知识 点分治 边分治 树链剖分 LCT Top Tree LCT时间复杂度 线段树每次查询是严格\(\log n\)的,然而splay维护连续段的时候,如果每次查 ...

  5. Loj #6073.「2017 山东一轮集训 Day5」距离

    Loj #6073.「2017 山东一轮集训 Day5」距离 Description 给定一棵 \(n\) 个点的边带权的树,以及一个排列$ p\(,有\)q $个询问,给定点 \(u, v, k\) ...

  6. 「2017 山东一轮集训 Day5」苹果树

    「2017 山东一轮集训 Day5」苹果树 \(n\leq 40\) 折半搜索+矩阵树定理. 没有想到折半搜索. 首先我们先枚举\(k\)个好点,我们让它们一定没有用的.要满足这个条件就要使它只能和坏 ...

  7. [2018HN省队集训D9T1] circle

    [2018HN省队集训D9T1] circle 题意 给定一个 \(n\) 个点的竞赛图并在其中钦定了 \(k\) 个点, 数据保证删去钦定的 \(k\) 个点后这个图没有环. 问在不删去钦定的这 \ ...

  8. [2018HN省队集训D8T1] 杀毒软件

    [2018HN省队集训D8T1] 杀毒软件 题意 给定一个 \(m\) 个01串的字典以及一个长度为 \(n\) 的 01? 序列. 对这个序列进行 \(q\) 次操作, 修改某个位置的字符情况以及查 ...

  9. [2018HN省队集训D8T3] 水果拼盘

    [2018HN省队集训D8T3] 水果拼盘 题意 给定 \(n\) 个集合, 每个集合包含 \([1,m]\) 中的一些整数, 在这些集合中随机选取 \(k\) 个集合, 求这 \(k\) 个集合的并 ...

随机推荐

  1. 第三次寒假作业 sketch 了解

    什么是sketch? sketch 是一种基于散列的数据结构,可以在高速网络环境中,实时地存储流量特征信息,只占用较小的空间资源,并且具备在理论上可证明的估计精度与内存的平衡特性. 通过设置散列函数, ...

  2. HDU 5794 A Simple Chess dp+Lucas

    题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=5794 A Simple Chess Time Limit: 2000/1000 MS (Java/O ...

  3. flash builder 4.6在debug调试时需要系统安装flashplayer debug版本

    http://blog.csdn.net/cupid0051/article/details/46684295

  4. win7 安装 MongoDB 及简单操作

    下载地址 http://dl.mongodb.org/dl/win32/x86_64 这里用的版本是 mongodb-latest-signed.msi 同时下载 mongodb-compass 下载 ...

  5. cacti 安装perl 和XML::Simple

    一.安装perl #tar zxvf perl-5.20.1.tar.gz #cd perl-5.20.1  #./Configure -de  #make  #make test  #make in ...

  6. Ubuntu 10.04 下载android 4.1.1_r4

    一.安装 curl git $ sudo apt-get install  curl $ sudo apt-get install git-core 二.安装repo 1.在主目录(~)建立目录 bi ...

  7. [BZOJ4942] [NOI2017]整数

    题目背景 在人类智慧的山巅,有着一台字长为1048576位(此数字与解题无关)的超级计算机,著名理论计算机科 学家P博士正用它进行各种研究.不幸的是,这天台风切断了电力系统,超级计算机 无法工作,而 ...

  8. Java第一次实验报告——Java开发环境的熟悉

    北京电子科技学院(BESTI) 实    验    报    告 课程名称:java程序设计实验      班级:1352         姓名:洪韶武      学号:20135219 成绩:   ...

  9. Codeforces 864E Fire(背包DP)

    背包DP,决策的时候记一下 jc[i][j]=1 表示第i个物品容量为j的时候要选,输出方案的时候倒推就好了 #include<iostream> #include<cstdlib& ...

  10. 【优先队列】【UVa11997】K Smallest Sums

    传送门 Description Input Output Translation · 给定k个长度为k的数组,把每个数组选一个元素加起来,这样共有kk种可能的答案,求最小的k个 Sample Inpu ...